From f47c50bf3a983fdad6ef3f1e19fbc6929c23e501 Mon Sep 17 00:00:00 2001 From: John Rofrano Date: Sun, 4 Feb 2024 16:01:58 +0000 Subject: [PATCH 1/5] Updated dev env with poetry --- .devcontainer/Dockerfile | 8 +- .devcontainer/devcontainer.json | 16 +- .gitignore | 1 + .pylintrc | 3 - .vscode/launch.json | 23 - .vscode/settings.json | 11 - Makefile | 9 +- poetry.lock | 1095 +++++++++++++++++++++++++++++++ pyproject.toml | 48 +- requirements.txt | 24 - setup.cfg | 18 +- 11 files changed, 1160 insertions(+), 96 deletions(-) delete mode 100644 .pylintrc delete mode 100644 .vscode/launch.json delete mode 100644 .vscode/settings.json create mode 100644 poetry.lock delete mode 100644 requirements.txt diff --git a/.devcontainer/Dockerfile b/.devcontainer/Dockerfile index 03a00ab..98fc4f5 100644 --- a/.devcontainer/Dockerfile +++ b/.devcontainer/Dockerfile @@ -1,5 +1,5 @@ # Image for a Python 3 development environment -FROM python:3.9-slim +FROM python:3.11-slim # Add any tools that are needed beyond Python RUN apt-get update && \ @@ -21,8 +21,10 @@ RUN groupadd --gid $USER_GID $USERNAME \ # Set up the Python development environment WORKDIR /app -RUN python -m pip install --upgrade pip && \ - pip install --upgrade wheel +COPY pyproject.toml poetry.lock ./ +RUN sudo python -m pip install --upgrade pip poetry && \ + sudo poetry config virtualenvs.create false && \ + sudo poetry install # Enable color terminal for docker exec bash ENV TERM=xterm-256color diff --git a/.devcontainer/devcontainer.json b/.devcontainer/devcontainer.json index a7b9940..f0e859e 100644 --- a/.devcontainer/devcontainer.json +++ b/.devcontainer/devcontainer.json @@ -1,10 +1,10 @@ +// cSpell: disable { "name": "Python 3 & Redis", "dockerComposeFile": "docker-compose.yml", "service": "app", "workspaceFolder": "/app", "remoteUser": "devops", - "customizations": { "vscode": { "settings": { @@ -14,11 +14,11 @@ }, "markdown-preview-github-styles.colorTheme": "light", "makefile.extensionOutputFolder": "/tmp", + "python.testing.unittestEnabled": false, + "python.testing.pytestEnabled": true, "python.testing.pytestArgs": [ "tests" ], - "python.testing.unittestEnabled": false, - "python.testing.pytestEnabled": true, "files.exclude": { "**/.git": true, "**/.DS_Store": true, @@ -33,18 +33,18 @@ "ms-python.pylint", "ms-python.flake8", "ms-python.black-formatter", - "njpwerner.autodocstring", - "wholroyd.jinja", "ms-vscode.makefile-tools", - "tamasfe.even-better-toml", "yzhang.markdown-all-in-one", "hnw.vscode-auto-open-markdown-preview", - "bierner.markdown-preview-github-styles", "davidanson.vscode-markdownlint", + "bierner.markdown-preview-github-styles", + "tamasfe.even-better-toml", "donjayamanne.githistory", "GitHub.vscode-pull-request-github", "hbenl.vscode-test-explorer", "LittleFoxTeam.vscode-python-test-adapter", + "njpwerner.autodocstring", + "wholroyd.jinja", "redhat.vscode-yaml", "ms-azuretools.vscode-docker", "ms-kubernetes-tools.vscode-kubernetes-tools", @@ -55,5 +55,5 @@ ] } }, - "postCreateCommand": "sudo pip install -r requirements.txt" + // "postCreateCommand": "bash /app/.devcontainer/scripts/post-install.sh" } diff --git a/.gitignore b/.gitignore index e751fb7..4296669 100644 --- a/.gitignore +++ b/.gitignore @@ -99,6 +99,7 @@ celerybeat-schedule .env # virtualenv +.venv/ venv/ ENV/ diff --git a/.pylintrc b/.pylintrc deleted file mode 100644 index c25f90e..0000000 --- a/.pylintrc +++ /dev/null @@ -1,3 +0,0 @@ -[MESSAGES CONTROL] -# Pylint cannot detect Flask app methods to we turn checkint off -disable=E1101 \ No newline at end of file diff --git a/.vscode/launch.json b/.vscode/launch.json deleted file mode 100644 index 66754d0..0000000 --- a/.vscode/launch.json +++ /dev/null @@ -1,23 +0,0 @@ -{ - // Use IntelliSense to learn about possible attributes. - // Hover to view descriptions of existing attributes. - // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 - "version": "0.2.0", - "configurations": [ - { - "name": "Python: Flask", - "type": "python", - "request": "launch", - "module": "flask", - "env": { - "FLASK_APP": "app.py", - "FLASK_ENV": "development" - }, - "args": [ - "run", - "--no-debugger" - ], - "jinja": true - } - ] -} \ No newline at end of file diff --git a/.vscode/settings.json b/.vscode/settings.json deleted file mode 100644 index 46c244d..0000000 --- a/.vscode/settings.json +++ /dev/null @@ -1,11 +0,0 @@ -{ - "files.exclude": { - "**/.git": true, - "**/.svn": true, - "**/.hg": true, - "**/CVS": true, - "**/.DS_Store": true, - "**/*.pyc": true, - "**/__pycache__": true - } -} diff --git a/Makefile b/Makefile index 21e9aa7..930608f 100644 --- a/Makefile +++ b/Makefile @@ -7,20 +7,21 @@ all: help venv: ## Create a Python virtual environment $(info Creating Python 3 virtual environment...) - python3 -m venv .venv + poetry shell install: ## Install dependencies $(info Installing dependencies...) - sudo pip install -r requirements.txt + sudo poetry install lint: ## Run the linter $(info Running linting...) flake8 service --count --select=E9,F63,F7,F82 --show-source --statistics flake8 service --count --max-complexity=10 --max-line-length=127 --statistics + pylint service --max-line-length=127 test: ## Run the unit tests - $(info Running tests...) - nosetests --with-spec --spec-color + $(info Running unit tests...) + pytest run: ## Run the service $(info Starting service...) diff --git a/poetry.lock b/poetry.lock new file mode 100644 index 0000000..2b5a210 --- /dev/null +++ b/poetry.lock @@ -0,0 +1,1095 @@ +# This file is automatically @generated by Poetry 1.7.1 and should not be changed by hand. + +[[package]] +name = "astroid" +version = "3.0.3" +description = "An abstract syntax tree for Python with inference support." +optional = false +python-versions = ">=3.8.0" +files = [ + {file = "astroid-3.0.3-py3-none-any.whl", hash = "sha256:92fcf218b89f449cdf9f7b39a269f8d5d617b27be68434912e11e79203963a17"}, + {file = "astroid-3.0.3.tar.gz", hash = "sha256:4148645659b08b70d72460ed1921158027a9e53ae8b7234149b1400eddacbb93"}, +] + +[[package]] +name = "async-timeout" +version = "4.0.3" +description = "Timeout context manager for asyncio programs" +optional = false +python-versions = ">=3.7" +files = [ + {file = "async-timeout-4.0.3.tar.gz", hash = "sha256:4640d96be84d82d02ed59ea2b7105a0f7b33abe8703703cd0ab0bf87c427522f"}, + {file = "async_timeout-4.0.3-py3-none-any.whl", hash = "sha256:7405140ff1230c310e51dc27b3145b9092d659ce68ff733fb0cefe3ee42be028"}, +] + +[[package]] +name = "black" +version = "23.12.1" +description = "The uncompromising code formatter." +optional = false +python-versions = ">=3.8" +files = [ + {file = "black-23.12.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:e0aaf6041986767a5e0ce663c7a2f0e9eaf21e6ff87a5f95cbf3675bfd4c41d2"}, + {file = "black-23.12.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:c88b3711d12905b74206227109272673edce0cb29f27e1385f33b0163c414bba"}, + {file = "black-23.12.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a920b569dc6b3472513ba6ddea21f440d4b4c699494d2e972a1753cdc25df7b0"}, + {file = "black-23.12.1-cp310-cp310-win_amd64.whl", hash = "sha256:3fa4be75ef2a6b96ea8d92b1587dd8cb3a35c7e3d51f0738ced0781c3aa3a5a3"}, + {file = "black-23.12.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:8d4df77958a622f9b5a4c96edb4b8c0034f8434032ab11077ec6c56ae9f384ba"}, + {file = "black-23.12.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:602cfb1196dc692424c70b6507593a2b29aac0547c1be9a1d1365f0d964c353b"}, + {file = "black-23.12.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9c4352800f14be5b4864016882cdba10755bd50805c95f728011bcb47a4afd59"}, + {file = "black-23.12.1-cp311-cp311-win_amd64.whl", hash = "sha256:0808494f2b2df923ffc5723ed3c7b096bd76341f6213989759287611e9837d50"}, + {file = "black-23.12.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:25e57fd232a6d6ff3f4478a6fd0580838e47c93c83eaf1ccc92d4faf27112c4e"}, + {file = "black-23.12.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:2d9e13db441c509a3763a7a3d9a49ccc1b4e974a47be4e08ade2a228876500ec"}, + {file = "black-23.12.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6d1bd9c210f8b109b1762ec9fd36592fdd528485aadb3f5849b2740ef17e674e"}, + {file = "black-23.12.1-cp312-cp312-win_amd64.whl", hash = "sha256:ae76c22bde5cbb6bfd211ec343ded2163bba7883c7bc77f6b756a1049436fbb9"}, + {file = "black-23.12.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1fa88a0f74e50e4487477bc0bb900c6781dbddfdfa32691e780bf854c3b4a47f"}, + {file = "black-23.12.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:a4d6a9668e45ad99d2f8ec70d5c8c04ef4f32f648ef39048d010b0689832ec6d"}, + {file = "black-23.12.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b18fb2ae6c4bb63eebe5be6bd869ba2f14fd0259bda7d18a46b764d8fb86298a"}, + {file = "black-23.12.1-cp38-cp38-win_amd64.whl", hash = "sha256:c04b6d9d20e9c13f43eee8ea87d44156b8505ca8a3c878773f68b4e4812a421e"}, + {file = "black-23.12.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:3e1b38b3135fd4c025c28c55ddfc236b05af657828a8a6abe5deec419a0b7055"}, + {file = "black-23.12.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4f0031eaa7b921db76decd73636ef3a12c942ed367d8c3841a0739412b260a54"}, + {file = "black-23.12.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:97e56155c6b737854e60a9ab1c598ff2533d57e7506d97af5481141671abf3ea"}, + {file = "black-23.12.1-cp39-cp39-win_amd64.whl", hash = "sha256:dd15245c8b68fe2b6bd0f32c1556509d11bb33aec9b5d0866dd8e2ed3dba09c2"}, + {file = "black-23.12.1-py3-none-any.whl", hash = "sha256:78baad24af0f033958cad29731e27363183e140962595def56423e626f4bee3e"}, + {file = "black-23.12.1.tar.gz", hash = "sha256:4ce3ef14ebe8d9509188014d96af1c456a910d5b5cbf434a09fef7e024b3d0d5"}, +] + +[package.dependencies] +click = ">=8.0.0" +mypy-extensions = ">=0.4.3" +packaging = ">=22.0" +pathspec = ">=0.9.0" +platformdirs = ">=2" + +[package.extras] +colorama = ["colorama (>=0.4.3)"] +d = ["aiohttp (>=3.7.4)", "aiohttp (>=3.7.4,!=3.9.0)"] +jupyter = ["ipython (>=7.8.0)", "tokenize-rt (>=3.2.0)"] +uvloop = ["uvloop (>=0.15.2)"] + +[[package]] +name = "blinker" +version = "1.7.0" +description = "Fast, simple object-to-object and broadcast signaling" +optional = false +python-versions = ">=3.8" +files = [ + {file = "blinker-1.7.0-py3-none-any.whl", hash = "sha256:c3f865d4d54db7abc53758a01601cf343fe55b84c1de4e3fa910e420b438d5b9"}, + {file = "blinker-1.7.0.tar.gz", hash = "sha256:e6820ff6fa4e4d1d8e2747c2283749c3f547e4fee112b98555cdcdae32996182"}, +] + +[[package]] +name = "certifi" +version = "2024.2.2" +description = "Python package for providing Mozilla's CA Bundle." +optional = false +python-versions = ">=3.6" +files = [ + {file = "certifi-2024.2.2-py3-none-any.whl", hash = "sha256:dc383c07b76109f368f6106eee2b593b04a011ea4d55f652c6ca24a754d1cdd1"}, + {file = "certifi-2024.2.2.tar.gz", hash = "sha256:0569859f95fc761b18b45ef421b1290a0f65f147e92a1e5eb3e635f9a5e4e66f"}, +] + +[[package]] +name = "charset-normalizer" +version = "3.3.2" +description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "charset-normalizer-3.3.2.tar.gz", hash = "sha256:f30c3cb33b24454a82faecaf01b19c18562b1e89558fb6c56de4d9118a032fd5"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:25baf083bf6f6b341f4121c2f3c548875ee6f5339300e08be3f2b2ba1721cdd3"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:06435b539f889b1f6f4ac1758871aae42dc3a8c0e24ac9e60c2384973ad73027"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9063e24fdb1e498ab71cb7419e24622516c4a04476b17a2dab57e8baa30d6e03"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6897af51655e3691ff853668779c7bad41579facacf5fd7253b0133308cf000d"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1d3193f4a680c64b4b6a9115943538edb896edc190f0b222e73761716519268e"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cd70574b12bb8a4d2aaa0094515df2463cb429d8536cfb6c7ce983246983e5a6"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8465322196c8b4d7ab6d1e049e4c5cb460d0394da4a27d23cc242fbf0034b6b5"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a9a8e9031d613fd2009c182b69c7b2c1ef8239a0efb1df3f7c8da66d5dd3d537"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:beb58fe5cdb101e3a055192ac291b7a21e3b7ef4f67fa1d74e331a7f2124341c"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:e06ed3eb3218bc64786f7db41917d4e686cc4856944f53d5bdf83a6884432e12"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:2e81c7b9c8979ce92ed306c249d46894776a909505d8f5a4ba55b14206e3222f"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:572c3763a264ba47b3cf708a44ce965d98555f618ca42c926a9c1616d8f34269"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:fd1abc0d89e30cc4e02e4064dc67fcc51bd941eb395c502aac3ec19fab46b519"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-win32.whl", hash = "sha256:3d47fa203a7bd9c5b6cee4736ee84ca03b8ef23193c0d1ca99b5089f72645c73"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-win_amd64.whl", hash = "sha256:10955842570876604d404661fbccbc9c7e684caf432c09c715ec38fbae45ae09"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:802fe99cca7457642125a8a88a084cef28ff0cf9407060f7b93dca5aa25480db"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:573f6eac48f4769d667c4442081b1794f52919e7edada77495aaed9236d13a96"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:549a3a73da901d5bc3ce8d24e0600d1fa85524c10287f6004fbab87672bf3e1e"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f27273b60488abe721a075bcca6d7f3964f9f6f067c8c4c605743023d7d3944f"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1ceae2f17a9c33cb48e3263960dc5fc8005351ee19db217e9b1bb15d28c02574"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:65f6f63034100ead094b8744b3b97965785388f308a64cf8d7c34f2f2e5be0c4"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:753f10e867343b4511128c6ed8c82f7bec3bd026875576dfd88483c5c73b2fd8"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4a78b2b446bd7c934f5dcedc588903fb2f5eec172f3d29e52a9096a43722adfc"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:e537484df0d8f426ce2afb2d0f8e1c3d0b114b83f8850e5f2fbea0e797bd82ae"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:eb6904c354526e758fda7167b33005998fb68c46fbc10e013ca97f21ca5c8887"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:deb6be0ac38ece9ba87dea880e438f25ca3eddfac8b002a2ec3d9183a454e8ae"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:4ab2fe47fae9e0f9dee8c04187ce5d09f48eabe611be8259444906793ab7cbce"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:80402cd6ee291dcb72644d6eac93785fe2c8b9cb30893c1af5b8fdd753b9d40f"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-win32.whl", hash = "sha256:7cd13a2e3ddeed6913a65e66e94b51d80a041145a026c27e6bb76c31a853c6ab"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-win_amd64.whl", hash = "sha256:663946639d296df6a2bb2aa51b60a2454ca1cb29835324c640dafb5ff2131a77"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:0b2b64d2bb6d3fb9112bafa732def486049e63de9618b5843bcdd081d8144cd8"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:ddbb2551d7e0102e7252db79ba445cdab71b26640817ab1e3e3648dad515003b"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:55086ee1064215781fff39a1af09518bc9255b50d6333f2e4c74ca09fac6a8f6"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8f4a014bc36d3c57402e2977dada34f9c12300af536839dc38c0beab8878f38a"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a10af20b82360ab00827f916a6058451b723b4e65030c5a18577c8b2de5b3389"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8d756e44e94489e49571086ef83b2bb8ce311e730092d2c34ca8f7d925cb20aa"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:90d558489962fd4918143277a773316e56c72da56ec7aa3dc3dbbe20fdfed15b"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6ac7ffc7ad6d040517be39eb591cac5ff87416c2537df6ba3cba3bae290c0fed"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:7ed9e526742851e8d5cc9e6cf41427dfc6068d4f5a3bb03659444b4cabf6bc26"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:8bdb58ff7ba23002a4c5808d608e4e6c687175724f54a5dade5fa8c67b604e4d"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:6b3251890fff30ee142c44144871185dbe13b11bab478a88887a639655be1068"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:b4a23f61ce87adf89be746c8a8974fe1c823c891d8f86eb218bb957c924bb143"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:efcb3f6676480691518c177e3b465bcddf57cea040302f9f4e6e191af91174d4"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-win32.whl", hash = "sha256:d965bba47ddeec8cd560687584e88cf699fd28f192ceb452d1d7ee807c5597b7"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-win_amd64.whl", hash = "sha256:96b02a3dc4381e5494fad39be677abcb5e6634bf7b4fa83a6dd3112607547001"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:95f2a5796329323b8f0512e09dbb7a1860c46a39da62ecb2324f116fa8fdc85c"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c002b4ffc0be611f0d9da932eb0f704fe2602a9a949d1f738e4c34c75b0863d5"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a981a536974bbc7a512cf44ed14938cf01030a99e9b3a06dd59578882f06f985"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3287761bc4ee9e33561a7e058c72ac0938c4f57fe49a09eae428fd88aafe7bb6"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:42cb296636fcc8b0644486d15c12376cb9fa75443e00fb25de0b8602e64c1714"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0a55554a2fa0d408816b3b5cedf0045f4b8e1a6065aec45849de2d6f3f8e9786"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:c083af607d2515612056a31f0a8d9e0fcb5876b7bfc0abad3ecd275bc4ebc2d5"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:87d1351268731db79e0f8e745d92493ee2841c974128ef629dc518b937d9194c"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:bd8f7df7d12c2db9fab40bdd87a7c09b1530128315d047a086fa3ae3435cb3a8"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:c180f51afb394e165eafe4ac2936a14bee3eb10debc9d9e4db8958fe36afe711"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:8c622a5fe39a48f78944a87d4fb8a53ee07344641b0562c540d840748571b811"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-win32.whl", hash = "sha256:db364eca23f876da6f9e16c9da0df51aa4f104a972735574842618b8c6d999d4"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-win_amd64.whl", hash = "sha256:86216b5cee4b06df986d214f664305142d9c76df9b6512be2738aa72a2048f99"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:6463effa3186ea09411d50efc7d85360b38d5f09b870c48e4600f63af490e56a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6c4caeef8fa63d06bd437cd4bdcf3ffefe6738fb1b25951440d80dc7df8c03ac"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:37e55c8e51c236f95b033f6fb391d7d7970ba5fe7ff453dad675e88cf303377a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fb69256e180cb6c8a894fee62b3afebae785babc1ee98b81cdf68bbca1987f33"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ae5f4161f18c61806f411a13b0310bea87f987c7d2ecdbdaad0e94eb2e404238"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b2b0a0c0517616b6869869f8c581d4eb2dd83a4d79e0ebcb7d373ef9956aeb0a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:45485e01ff4d3630ec0d9617310448a8702f70e9c01906b0d0118bdf9d124cf2"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:eb00ed941194665c332bf8e078baf037d6c35d7c4f3102ea2d4f16ca94a26dc8"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:2127566c664442652f024c837091890cb1942c30937add288223dc895793f898"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:a50aebfa173e157099939b17f18600f72f84eed3049e743b68ad15bd69b6bf99"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:4d0d1650369165a14e14e1e47b372cfcb31d6ab44e6e33cb2d4e57265290044d"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:923c0c831b7cfcb071580d3f46c4baf50f174be571576556269530f4bbd79d04"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:06a81e93cd441c56a9b65d8e1d043daeb97a3d0856d177d5c90ba85acb3db087"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-win32.whl", hash = "sha256:6ef1d82a3af9d3eecdba2321dc1b3c238245d890843e040e41e470ffa64c3e25"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-win_amd64.whl", hash = "sha256:eb8821e09e916165e160797a6c17edda0679379a4be5c716c260e836e122f54b"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:c235ebd9baae02f1b77bcea61bce332cb4331dc3617d254df3323aa01ab47bd4"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5b4c145409bef602a690e7cfad0a15a55c13320ff7a3ad7ca59c13bb8ba4d45d"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:68d1f8a9e9e37c1223b656399be5d6b448dea850bed7d0f87a8311f1ff3dabb0"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:22afcb9f253dac0696b5a4be4a1c0f8762f8239e21b99680099abd9b2b1b2269"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e27ad930a842b4c5eb8ac0016b0a54f5aebbe679340c26101df33424142c143c"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1f79682fbe303db92bc2b1136016a38a42e835d932bab5b3b1bfcfbf0640e519"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b261ccdec7821281dade748d088bb6e9b69e6d15b30652b74cbbac25e280b796"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:122c7fa62b130ed55f8f285bfd56d5f4b4a5b503609d181f9ad85e55c89f4185"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:d0eccceffcb53201b5bfebb52600a5fb483a20b61da9dbc885f8b103cbe7598c"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:9f96df6923e21816da7e0ad3fd47dd8f94b2a5ce594e00677c0013018b813458"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:7f04c839ed0b6b98b1a7501a002144b76c18fb1c1850c8b98d458ac269e26ed2"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:34d1c8da1e78d2e001f363791c98a272bb734000fcef47a491c1e3b0505657a8"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ff8fa367d09b717b2a17a052544193ad76cd49979c805768879cb63d9ca50561"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-win32.whl", hash = "sha256:aed38f6e4fb3f5d6bf81bfa990a07806be9d83cf7bacef998ab1a9bd660a581f"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-win_amd64.whl", hash = "sha256:b01b88d45a6fcb69667cd6d2f7a9aeb4bf53760d7fc536bf679ec94fe9f3ff3d"}, + {file = "charset_normalizer-3.3.2-py3-none-any.whl", hash = "sha256:3e4d1f6587322d2788836a99c69062fbb091331ec940e02d12d179c1d53e25fc"}, +] + +[[package]] +name = "click" +version = "8.1.7" +description = "Composable command line interface toolkit" +optional = false +python-versions = ">=3.7" +files = [ + {file = "click-8.1.7-py3-none-any.whl", hash = "sha256:ae74fb96c20a0277a1d615f1e4d73c8414f5a98db8b799a7931d1582f3390c28"}, + {file = "click-8.1.7.tar.gz", hash = "sha256:ca9853ad459e787e2192211578cc907e7594e294c7ccc834310722b41b9ca6de"}, +] + +[package.dependencies] +colorama = {version = "*", markers = "platform_system == \"Windows\""} + +[[package]] +name = "colorama" +version = "0.4.6" +description = "Cross-platform colored terminal text." +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7" +files = [ + {file = "colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6"}, + {file = "colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44"}, +] + +[[package]] +name = "coverage" +version = "7.4.1" +description = "Code coverage measurement for Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "coverage-7.4.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:077d366e724f24fc02dbfe9d946534357fda71af9764ff99d73c3c596001bbd7"}, + {file = "coverage-7.4.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:0193657651f5399d433c92f8ae264aff31fc1d066deee4b831549526433f3f61"}, + {file = "coverage-7.4.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d17bbc946f52ca67adf72a5ee783cd7cd3477f8f8796f59b4974a9b59cacc9ee"}, + {file = "coverage-7.4.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a3277f5fa7483c927fe3a7b017b39351610265308f5267ac6d4c2b64cc1d8d25"}, + {file = "coverage-7.4.1-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6dceb61d40cbfcf45f51e59933c784a50846dc03211054bd76b421a713dcdf19"}, + {file = "coverage-7.4.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:6008adeca04a445ea6ef31b2cbaf1d01d02986047606f7da266629afee982630"}, + {file = "coverage-7.4.1-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:c61f66d93d712f6e03369b6a7769233bfda880b12f417eefdd4f16d1deb2fc4c"}, + {file = "coverage-7.4.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:b9bb62fac84d5f2ff523304e59e5c439955fb3b7f44e3d7b2085184db74d733b"}, + {file = "coverage-7.4.1-cp310-cp310-win32.whl", hash = "sha256:f86f368e1c7ce897bf2457b9eb61169a44e2ef797099fb5728482b8d69f3f016"}, + {file = "coverage-7.4.1-cp310-cp310-win_amd64.whl", hash = "sha256:869b5046d41abfea3e381dd143407b0d29b8282a904a19cb908fa24d090cc018"}, + {file = "coverage-7.4.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:b8ffb498a83d7e0305968289441914154fb0ef5d8b3157df02a90c6695978295"}, + {file = "coverage-7.4.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:3cacfaefe6089d477264001f90f55b7881ba615953414999c46cc9713ff93c8c"}, + {file = "coverage-7.4.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5d6850e6e36e332d5511a48a251790ddc545e16e8beaf046c03985c69ccb2676"}, + {file = "coverage-7.4.1-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:18e961aa13b6d47f758cc5879383d27b5b3f3dcd9ce8cdbfdc2571fe86feb4dd"}, + {file = "coverage-7.4.1-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dfd1e1b9f0898817babf840b77ce9fe655ecbe8b1b327983df485b30df8cc011"}, + {file = "coverage-7.4.1-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:6b00e21f86598b6330f0019b40fb397e705135040dbedc2ca9a93c7441178e74"}, + {file = "coverage-7.4.1-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:536d609c6963c50055bab766d9951b6c394759190d03311f3e9fcf194ca909e1"}, + {file = "coverage-7.4.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:7ac8f8eb153724f84885a1374999b7e45734bf93a87d8df1e7ce2146860edef6"}, + {file = "coverage-7.4.1-cp311-cp311-win32.whl", hash = "sha256:f3771b23bb3675a06f5d885c3630b1d01ea6cac9e84a01aaf5508706dba546c5"}, + {file = "coverage-7.4.1-cp311-cp311-win_amd64.whl", hash = "sha256:9d2f9d4cc2a53b38cabc2d6d80f7f9b7e3da26b2f53d48f05876fef7956b6968"}, + {file = "coverage-7.4.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:f68ef3660677e6624c8cace943e4765545f8191313a07288a53d3da188bd8581"}, + {file = "coverage-7.4.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:23b27b8a698e749b61809fb637eb98ebf0e505710ec46a8aa6f1be7dc0dc43a6"}, + {file = "coverage-7.4.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3e3424c554391dc9ef4a92ad28665756566a28fecf47308f91841f6c49288e66"}, + {file = "coverage-7.4.1-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e0860a348bf7004c812c8368d1fc7f77fe8e4c095d661a579196a9533778e156"}, + {file = "coverage-7.4.1-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fe558371c1bdf3b8fa03e097c523fb9645b8730399c14fe7721ee9c9e2a545d3"}, + {file = "coverage-7.4.1-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:3468cc8720402af37b6c6e7e2a9cdb9f6c16c728638a2ebc768ba1ef6f26c3a1"}, + {file = "coverage-7.4.1-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:02f2edb575d62172aa28fe00efe821ae31f25dc3d589055b3fb64d51e52e4ab1"}, + {file = "coverage-7.4.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:ca6e61dc52f601d1d224526360cdeab0d0712ec104a2ce6cc5ccef6ed9a233bc"}, + {file = "coverage-7.4.1-cp312-cp312-win32.whl", hash = "sha256:ca7b26a5e456a843b9b6683eada193fc1f65c761b3a473941efe5a291f604c74"}, + {file = "coverage-7.4.1-cp312-cp312-win_amd64.whl", hash = "sha256:85ccc5fa54c2ed64bd91ed3b4a627b9cce04646a659512a051fa82a92c04a448"}, + {file = "coverage-7.4.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:8bdb0285a0202888d19ec6b6d23d5990410decb932b709f2b0dfe216d031d218"}, + {file = "coverage-7.4.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:918440dea04521f499721c039863ef95433314b1db00ff826a02580c1f503e45"}, + {file = "coverage-7.4.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:379d4c7abad5afbe9d88cc31ea8ca262296480a86af945b08214eb1a556a3e4d"}, + {file = "coverage-7.4.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b094116f0b6155e36a304ff912f89bbb5067157aff5f94060ff20bbabdc8da06"}, + {file = "coverage-7.4.1-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f2f5968608b1fe2a1d00d01ad1017ee27efd99b3437e08b83ded9b7af3f6f766"}, + {file = "coverage-7.4.1-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:10e88e7f41e6197ea0429ae18f21ff521d4f4490aa33048f6c6f94c6045a6a75"}, + {file = "coverage-7.4.1-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:a4a3907011d39dbc3e37bdc5df0a8c93853c369039b59efa33a7b6669de04c60"}, + {file = "coverage-7.4.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:6d224f0c4c9c98290a6990259073f496fcec1b5cc613eecbd22786d398ded3ad"}, + {file = "coverage-7.4.1-cp38-cp38-win32.whl", hash = "sha256:23f5881362dcb0e1a92b84b3c2809bdc90db892332daab81ad8f642d8ed55042"}, + {file = "coverage-7.4.1-cp38-cp38-win_amd64.whl", hash = "sha256:a07f61fc452c43cd5328b392e52555f7d1952400a1ad09086c4a8addccbd138d"}, + {file = "coverage-7.4.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:8e738a492b6221f8dcf281b67129510835461132b03024830ac0e554311a5c54"}, + {file = "coverage-7.4.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:46342fed0fff72efcda77040b14728049200cbba1279e0bf1188f1f2078c1d70"}, + {file = "coverage-7.4.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9641e21670c68c7e57d2053ddf6c443e4f0a6e18e547e86af3fad0795414a628"}, + {file = "coverage-7.4.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:aeb2c2688ed93b027eb0d26aa188ada34acb22dceea256d76390eea135083950"}, + {file = "coverage-7.4.1-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d12c923757de24e4e2110cf8832d83a886a4cf215c6e61ed506006872b43a6d1"}, + {file = "coverage-7.4.1-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:0491275c3b9971cdbd28a4595c2cb5838f08036bca31765bad5e17edf900b2c7"}, + {file = "coverage-7.4.1-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:8dfc5e195bbef80aabd81596ef52a1277ee7143fe419efc3c4d8ba2754671756"}, + {file = "coverage-7.4.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:1a78b656a4d12b0490ca72651fe4d9f5e07e3c6461063a9b6265ee45eb2bdd35"}, + {file = "coverage-7.4.1-cp39-cp39-win32.whl", hash = "sha256:f90515974b39f4dea2f27c0959688621b46d96d5a626cf9c53dbc653a895c05c"}, + {file = "coverage-7.4.1-cp39-cp39-win_amd64.whl", hash = "sha256:64e723ca82a84053dd7bfcc986bdb34af8d9da83c521c19d6b472bc6880e191a"}, + {file = "coverage-7.4.1-pp38.pp39.pp310-none-any.whl", hash = "sha256:32a8d985462e37cfdab611a6f95b09d7c091d07668fdc26e47a725ee575fe166"}, + {file = "coverage-7.4.1.tar.gz", hash = "sha256:1ed4b95480952b1a26d863e546fa5094564aa0065e1e5f0d4d0041f293251d04"}, +] + +[package.extras] +toml = ["tomli"] + +[[package]] +name = "defusedxml" +version = "0.7.1" +description = "XML bomb protection for Python stdlib modules" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +files = [ + {file = "defusedxml-0.7.1-py2.py3-none-any.whl", hash = "sha256:a352e7e428770286cc899e2542b6cdaedb2b4953ff269a210103ec58f6198a61"}, + {file = "defusedxml-0.7.1.tar.gz", hash = "sha256:1bb3032db185915b62d7c6209c5a8792be6a32ab2fedacc84e01b52c51aa3e69"}, +] + +[[package]] +name = "dill" +version = "0.3.8" +description = "serialize all of Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "dill-0.3.8-py3-none-any.whl", hash = "sha256:c36ca9ffb54365bdd2f8eb3eff7d2a21237f8452b57ace88b1ac615b7e815bd7"}, + {file = "dill-0.3.8.tar.gz", hash = "sha256:3ebe3c479ad625c4553aca177444d89b486b1d84982eeacded644afc0cf797ca"}, +] + +[package.extras] +graph = ["objgraph (>=1.7.2)"] +profile = ["gprof2dot (>=2022.7.29)"] + +[[package]] +name = "factory-boy" +version = "3.3.0" +description = "A versatile test fixtures replacement based on thoughtbot's factory_bot for Ruby." +optional = false +python-versions = ">=3.7" +files = [ + {file = "factory_boy-3.3.0-py2.py3-none-any.whl", hash = "sha256:a2cdbdb63228177aa4f1c52f4b6d83fab2b8623bf602c7dedd7eb83c0f69c04c"}, + {file = "factory_boy-3.3.0.tar.gz", hash = "sha256:bc76d97d1a65bbd9842a6d722882098eb549ec8ee1081f9fb2e8ff29f0c300f1"}, +] + +[package.dependencies] +Faker = ">=0.7.0" + +[package.extras] +dev = ["Django", "Pillow", "SQLAlchemy", "coverage", "flake8", "isort", "mongoengine", "sqlalchemy-utils", "tox", "wheel (>=0.32.0)", "zest.releaser[recommended]"] +doc = ["Sphinx", "sphinx-rtd-theme", "sphinxcontrib-spelling"] + +[[package]] +name = "faker" +version = "22.6.0" +description = "Faker is a Python package that generates fake data for you." +optional = false +python-versions = ">=3.8" +files = [ + {file = "Faker-22.6.0-py3-none-any.whl", hash = "sha256:2b57f0256da6b45b7851dca87836ef5e2ae2fbb64d63d8697f1e47830d7b505d"}, + {file = "Faker-22.6.0.tar.gz", hash = "sha256:fa6d969728ef3da6229da91267a1bd4e6b902044c4822012d4fc46c71bb92b26"}, +] + +[package.dependencies] +python-dateutil = ">=2.4" + +[[package]] +name = "flake8" +version = "6.1.0" +description = "the modular source code checker: pep8 pyflakes and co" +optional = false +python-versions = ">=3.8.1" +files = [ + {file = "flake8-6.1.0-py2.py3-none-any.whl", hash = "sha256:ffdfce58ea94c6580c77888a86506937f9a1a227dfcd15f245d694ae20a6b6e5"}, + {file = "flake8-6.1.0.tar.gz", hash = "sha256:d5b3857f07c030bdb5bf41c7f53799571d75c4491748a3adcd47de929e34cd23"}, +] + +[package.dependencies] +mccabe = ">=0.7.0,<0.8.0" +pycodestyle = ">=2.11.0,<2.12.0" +pyflakes = ">=3.1.0,<3.2.0" + +[[package]] +name = "flask" +version = "3.0.2" +description = "A simple framework for building complex web applications." +optional = false +python-versions = ">=3.8" +files = [ + {file = "flask-3.0.2-py3-none-any.whl", hash = "sha256:3232e0e9c850d781933cf0207523d1ece087eb8d87b23777ae38456e2fbe7c6e"}, + {file = "flask-3.0.2.tar.gz", hash = "sha256:822c03f4b799204250a7ee84b1eddc40665395333973dfb9deebfe425fefcb7d"}, +] + +[package.dependencies] +blinker = ">=1.6.2" +click = ">=8.1.3" +itsdangerous = ">=2.1.2" +Jinja2 = ">=3.1.2" +Werkzeug = ">=3.0.0" + +[package.extras] +async = ["asgiref (>=3.2)"] +dotenv = ["python-dotenv"] + +[[package]] +name = "gunicorn" +version = "21.2.0" +description = "WSGI HTTP Server for UNIX" +optional = false +python-versions = ">=3.5" +files = [ + {file = "gunicorn-21.2.0-py3-none-any.whl", hash = "sha256:3213aa5e8c24949e792bcacfc176fef362e7aac80b76c56f6b5122bf350722f0"}, + {file = "gunicorn-21.2.0.tar.gz", hash = "sha256:88ec8bff1d634f98e61b9f65bc4bf3cd918a90806c6f5c48bc5603849ec81033"}, +] + +[package.dependencies] +packaging = "*" + +[package.extras] +eventlet = ["eventlet (>=0.24.1)"] +gevent = ["gevent (>=1.4.0)"] +setproctitle = ["setproctitle"] +tornado = ["tornado (>=0.2)"] + +[[package]] +name = "honcho" +version = "1.1.0" +description = "Honcho: a Python clone of Foreman. For managing Procfile-based applications." +optional = false +python-versions = "*" +files = [ + {file = "honcho-1.1.0-py2.py3-none-any.whl", hash = "sha256:a4d6e3a88a7b51b66351ecfc6e9d79d8f4b87351db9ad7e923f5632cc498122f"}, + {file = "honcho-1.1.0.tar.gz", hash = "sha256:c5eca0bded4bef6697a23aec0422fd4f6508ea3581979a3485fc4b89357eb2a9"}, +] + +[package.dependencies] +colorama = {version = "*", markers = "sys_platform == \"win32\""} + +[package.extras] +export = ["jinja2 (>=2.7,<3)"] + +[[package]] +name = "httpie" +version = "3.2.2" +description = "HTTPie: modern, user-friendly command-line HTTP client for the API era." +optional = false +python-versions = ">=3.7" +files = [ + {file = "httpie-3.2.2-py3-none-any.whl", hash = "sha256:33082092553126e3e38395682c23121c3450472ffddb363eb895ad685d01800e"}, + {file = "httpie-3.2.2.tar.gz", hash = "sha256:8bfb671f0b39505c197fdef3367f7f99af5d0e81a4e22289bb4c1f0e72251c90"}, +] + +[package.dependencies] +charset-normalizer = ">=2.0.0" +colorama = {version = ">=0.2.4", markers = "sys_platform == \"win32\""} +defusedxml = ">=0.6.0" +multidict = ">=4.7.0" +pip = "*" +Pygments = ">=2.5.2" +requests = {version = ">=2.22.0", extras = ["socks"]} +requests-toolbelt = ">=0.9.1" +rich = ">=9.10.0" +setuptools = "*" + +[package.extras] +dev = ["Jinja2", "flake8", "flake8-comprehensions", "flake8-deprecated", "flake8-mutable", "flake8-tuple", "pyopenssl", "pytest", "pytest-cov", "pytest-httpbin (>=0.0.6)", "pytest-lazy-fixture (>=0.0.6)", "pytest-mock", "pyyaml", "responses", "twine", "werkzeug (<2.1.0)", "wheel"] +test = ["pytest", "pytest-httpbin (>=0.0.6)", "pytest-lazy-fixture (>=0.0.6)", "pytest-mock", "responses", "werkzeug (<2.1.0)"] + +[[package]] +name = "idna" +version = "3.6" +description = "Internationalized Domain Names in Applications (IDNA)" +optional = false +python-versions = ">=3.5" +files = [ + {file = "idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f"}, + {file = "idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca"}, +] + +[[package]] +name = "iniconfig" +version = "2.0.0" +description = "brain-dead simple config-ini parsing" +optional = false +python-versions = ">=3.7" +files = [ + {file = "iniconfig-2.0.0-py3-none-any.whl", hash = "sha256:b6a85871a79d2e3b22d2d1b94ac2824226a63c6b741c88f7ae975f18b6778374"}, + {file = "iniconfig-2.0.0.tar.gz", hash = "sha256:2d91e135bf72d31a410b17c16da610a82cb55f6b0477d1a902134b24a455b8b3"}, +] + +[[package]] +name = "isort" +version = "5.13.2" +description = "A Python utility / library to sort Python imports." +optional = false +python-versions = ">=3.8.0" +files = [ + {file = "isort-5.13.2-py3-none-any.whl", hash = "sha256:8ca5e72a8d85860d5a3fa69b8745237f2939afe12dbf656afbcb47fe72d947a6"}, + {file = "isort-5.13.2.tar.gz", hash = "sha256:48fdfcb9face5d58a4f6dde2e72a1fb8dcaf8ab26f95ab49fab84c2ddefb0109"}, +] + +[package.extras] +colors = ["colorama (>=0.4.6)"] + +[[package]] +name = "itsdangerous" +version = "2.1.2" +description = "Safely pass data to untrusted environments and back." +optional = false +python-versions = ">=3.7" +files = [ + {file = "itsdangerous-2.1.2-py3-none-any.whl", hash = "sha256:2c2349112351b88699d8d4b6b075022c0808887cb7ad10069318a8b0bc88db44"}, + {file = "itsdangerous-2.1.2.tar.gz", hash = "sha256:5dbbc68b317e5e42f327f9021763545dc3fc3bfe22e6deb96aaf1fc38874156a"}, +] + +[[package]] +name = "jinja2" +version = "3.1.3" +description = "A very fast and expressive template engine." +optional = false +python-versions = ">=3.7" +files = [ + {file = "Jinja2-3.1.3-py3-none-any.whl", hash = "sha256:7d6d50dd97d52cbc355597bd845fabfbac3f551e1f99619e39a35ce8c370b5fa"}, + {file = "Jinja2-3.1.3.tar.gz", hash = "sha256:ac8bd6544d4bb2c9792bf3a159e80bba8fda7f07e81bc3aed565432d5925ba90"}, +] + +[package.dependencies] +MarkupSafe = ">=2.0" + +[package.extras] +i18n = ["Babel (>=2.7)"] + +[[package]] +name = "markdown-it-py" +version = "3.0.0" +description = "Python port of markdown-it. Markdown parsing, done right!" +optional = false +python-versions = ">=3.8" +files = [ + {file = "markdown-it-py-3.0.0.tar.gz", hash = "sha256:e3f60a94fa066dc52ec76661e37c851cb232d92f9886b15cb560aaada2df8feb"}, + {file = "markdown_it_py-3.0.0-py3-none-any.whl", hash = "sha256:355216845c60bd96232cd8d8c40e8f9765cc86f46880e43a8fd22dc1a1a8cab1"}, +] + +[package.dependencies] +mdurl = ">=0.1,<1.0" + +[package.extras] +benchmarking = ["psutil", "pytest", "pytest-benchmark"] +code-style = ["pre-commit (>=3.0,<4.0)"] +compare = ["commonmark (>=0.9,<1.0)", "markdown (>=3.4,<4.0)", "mistletoe (>=1.0,<2.0)", "mistune (>=2.0,<3.0)", "panflute (>=2.3,<3.0)"] +linkify = ["linkify-it-py (>=1,<3)"] +plugins = ["mdit-py-plugins"] +profiling = ["gprof2dot"] +rtd = ["jupyter_sphinx", "mdit-py-plugins", "myst-parser", "pyyaml", "sphinx", "sphinx-copybutton", "sphinx-design", "sphinx_book_theme"] +testing = ["coverage", "pytest", "pytest-cov", "pytest-regressions"] + +[[package]] +name = "markupsafe" +version = "2.1.5" +description = "Safely add untrusted strings to HTML/XML markup." +optional = false +python-versions = ">=3.7" +files = [ + {file = "MarkupSafe-2.1.5-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:a17a92de5231666cfbe003f0e4b9b3a7ae3afb1ec2845aadc2bacc93ff85febc"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:72b6be590cc35924b02c78ef34b467da4ba07e4e0f0454a2c5907f473fc50ce5"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e61659ba32cf2cf1481e575d0462554625196a1f2fc06a1c777d3f48e8865d46"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2174c595a0d73a3080ca3257b40096db99799265e1c27cc5a610743acd86d62f"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ae2ad8ae6ebee9d2d94b17fb62763125f3f374c25618198f40cbb8b525411900"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:075202fa5b72c86ad32dc7d0b56024ebdbcf2048c0ba09f1cde31bfdd57bcfff"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:598e3276b64aff0e7b3451b72e94fa3c238d452e7ddcd893c3ab324717456bad"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:fce659a462a1be54d2ffcacea5e3ba2d74daa74f30f5f143fe0c58636e355fdd"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-win32.whl", hash = "sha256:d9fad5155d72433c921b782e58892377c44bd6252b5af2f67f16b194987338a4"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-win_amd64.whl", hash = "sha256:bf50cd79a75d181c9181df03572cdce0fbb75cc353bc350712073108cba98de5"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:629ddd2ca402ae6dbedfceeba9c46d5f7b2a61d9749597d4307f943ef198fc1f"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:5b7b716f97b52c5a14bffdf688f971b2d5ef4029127f1ad7a513973cfd818df2"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6ec585f69cec0aa07d945b20805be741395e28ac1627333b1c5b0105962ffced"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b91c037585eba9095565a3556f611e3cbfaa42ca1e865f7b8015fe5c7336d5a5"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7502934a33b54030eaf1194c21c692a534196063db72176b0c4028e140f8f32c"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:0e397ac966fdf721b2c528cf028494e86172b4feba51d65f81ffd65c63798f3f"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:c061bb86a71b42465156a3ee7bd58c8c2ceacdbeb95d05a99893e08b8467359a"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:3a57fdd7ce31c7ff06cdfbf31dafa96cc533c21e443d57f5b1ecc6cdc668ec7f"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-win32.whl", hash = "sha256:397081c1a0bfb5124355710fe79478cdbeb39626492b15d399526ae53422b906"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-win_amd64.whl", hash = "sha256:2b7c57a4dfc4f16f7142221afe5ba4e093e09e728ca65c51f5620c9aaeb9a617"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:8dec4936e9c3100156f8a2dc89c4b88d5c435175ff03413b443469c7c8c5f4d1"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:3c6b973f22eb18a789b1460b4b91bf04ae3f0c4234a0a6aa6b0a92f6f7b951d4"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ac07bad82163452a6884fe8fa0963fb98c2346ba78d779ec06bd7a6262132aee"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f5dfb42c4604dddc8e4305050aa6deb084540643ed5804d7455b5df8fe16f5e5"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ea3d8a3d18833cf4304cd2fc9cbb1efe188ca9b5efef2bdac7adc20594a0e46b"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:d050b3361367a06d752db6ead6e7edeb0009be66bc3bae0ee9d97fb326badc2a"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:bec0a414d016ac1a18862a519e54b2fd0fc8bbfd6890376898a6c0891dd82e9f"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:58c98fee265677f63a4385256a6d7683ab1832f3ddd1e66fe948d5880c21a169"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-win32.whl", hash = "sha256:8590b4ae07a35970728874632fed7bd57b26b0102df2d2b233b6d9d82f6c62ad"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-win_amd64.whl", hash = "sha256:823b65d8706e32ad2df51ed89496147a42a2a6e01c13cfb6ffb8b1e92bc910bb"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:c8b29db45f8fe46ad280a7294f5c3ec36dbac9491f2d1c17345be8e69cc5928f"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ec6a563cff360b50eed26f13adc43e61bc0c04d94b8be985e6fb24b81f6dcfdf"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a549b9c31bec33820e885335b451286e2969a2d9e24879f83fe904a5ce59d70a"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4f11aa001c540f62c6166c7726f71f7573b52c68c31f014c25cc7901deea0b52"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:7b2e5a267c855eea6b4283940daa6e88a285f5f2a67f2220203786dfa59b37e9"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:2d2d793e36e230fd32babe143b04cec8a8b3eb8a3122d2aceb4a371e6b09b8df"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:ce409136744f6521e39fd8e2a24c53fa18ad67aa5bc7c2cf83645cce5b5c4e50"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-win32.whl", hash = "sha256:4096e9de5c6fdf43fb4f04c26fb114f61ef0bf2e5604b6ee3019d51b69e8c371"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-win_amd64.whl", hash = "sha256:4275d846e41ecefa46e2015117a9f491e57a71ddd59bbead77e904dc02b1bed2"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:656f7526c69fac7f600bd1f400991cc282b417d17539a1b228617081106feb4a"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:97cafb1f3cbcd3fd2b6fbfb99ae11cdb14deea0736fc2b0952ee177f2b813a46"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1f3fbcb7ef1f16e48246f704ab79d79da8a46891e2da03f8783a5b6fa41a9532"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fa9db3f79de01457b03d4f01b34cf91bc0048eb2c3846ff26f66687c2f6d16ab"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ffee1f21e5ef0d712f9033568f8344d5da8cc2869dbd08d87c84656e6a2d2f68"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:5dedb4db619ba5a2787a94d877bc8ffc0566f92a01c0ef214865e54ecc9ee5e0"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:30b600cf0a7ac9234b2638fbc0fb6158ba5bdcdf46aeb631ead21248b9affbc4"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:8dd717634f5a044f860435c1d8c16a270ddf0ef8588d4887037c5028b859b0c3"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-win32.whl", hash = "sha256:daa4ee5a243f0f20d528d939d06670a298dd39b1ad5f8a72a4275124a7819eff"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-win_amd64.whl", hash = "sha256:619bc166c4f2de5caa5a633b8b7326fbe98e0ccbfacabd87268a2b15ff73a029"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:7a68b554d356a91cce1236aa7682dc01df0edba8d043fd1ce607c49dd3c1edcf"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:db0b55e0f3cc0be60c1f19efdde9a637c32740486004f20d1cff53c3c0ece4d2"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3e53af139f8579a6d5f7b76549125f0d94d7e630761a2111bc431fd820e163b8"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:17b950fccb810b3293638215058e432159d2b71005c74371d784862b7e4683f3"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4c31f53cdae6ecfa91a77820e8b151dba54ab528ba65dfd235c80b086d68a465"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:bff1b4290a66b490a2f4719358c0cdcd9bafb6b8f061e45c7a2460866bf50c2e"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:bc1667f8b83f48511b94671e0e441401371dfd0f0a795c7daa4a3cd1dde55bea"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:5049256f536511ee3f7e1b3f87d1d1209d327e818e6ae1365e8653d7e3abb6a6"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-win32.whl", hash = "sha256:00e046b6dd71aa03a41079792f8473dc494d564611a8f89bbbd7cb93295ebdcf"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-win_amd64.whl", hash = "sha256:fa173ec60341d6bb97a89f5ea19c85c5643c1e7dedebc22f5181eb73573142c5"}, + {file = "MarkupSafe-2.1.5.tar.gz", hash = "sha256:d283d37a890ba4c1ae73ffadf8046435c76e7bc2247bbb63c00bd1a709c6544b"}, +] + +[[package]] +name = "mccabe" +version = "0.7.0" +description = "McCabe checker, plugin for flake8" +optional = false +python-versions = ">=3.6" +files = [ + {file = "mccabe-0.7.0-py2.py3-none-any.whl", hash = "sha256:6c2d30ab6be0e4a46919781807b4f0d834ebdd6c6e3dca0bda5a15f863427b6e"}, + {file = "mccabe-0.7.0.tar.gz", hash = "sha256:348e0240c33b60bbdf4e523192ef919f28cb2c3d7d5c7794f74009290f236325"}, +] + +[[package]] +name = "mdurl" +version = "0.1.2" +description = "Markdown URL utilities" +optional = false +python-versions = ">=3.7" +files = [ + {file = "mdurl-0.1.2-py3-none-any.whl", hash = "sha256:84008a41e51615a49fc9966191ff91509e3c40b939176e643fd50a5c2196b8f8"}, + {file = "mdurl-0.1.2.tar.gz", hash = "sha256:bb413d29f5eea38f31dd4754dd7377d4465116fb207585f97bf925588687c1ba"}, +] + +[[package]] +name = "multidict" +version = "6.0.5" +description = "multidict implementation" +optional = false +python-versions = ">=3.7" +files = [ + {file = "multidict-6.0.5-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:228b644ae063c10e7f324ab1ab6b548bdf6f8b47f3ec234fef1093bc2735e5f9"}, + {file = "multidict-6.0.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:896ebdcf62683551312c30e20614305f53125750803b614e9e6ce74a96232604"}, + {file = "multidict-6.0.5-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:411bf8515f3be9813d06004cac41ccf7d1cd46dfe233705933dd163b60e37600"}, + {file = "multidict-6.0.5-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1d147090048129ce3c453f0292e7697d333db95e52616b3793922945804a433c"}, + {file = "multidict-6.0.5-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:215ed703caf15f578dca76ee6f6b21b7603791ae090fbf1ef9d865571039ade5"}, + {file = "multidict-6.0.5-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:7c6390cf87ff6234643428991b7359b5f59cc15155695deb4eda5c777d2b880f"}, + {file = "multidict-6.0.5-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:21fd81c4ebdb4f214161be351eb5bcf385426bf023041da2fd9e60681f3cebae"}, + {file = "multidict-6.0.5-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3cc2ad10255f903656017363cd59436f2111443a76f996584d1077e43ee51182"}, + {file = "multidict-6.0.5-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:6939c95381e003f54cd4c5516740faba40cf5ad3eeff460c3ad1d3e0ea2549bf"}, + {file = "multidict-6.0.5-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:220dd781e3f7af2c2c1053da9fa96d9cf3072ca58f057f4c5adaaa1cab8fc442"}, + {file = "multidict-6.0.5-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:766c8f7511df26d9f11cd3a8be623e59cca73d44643abab3f8c8c07620524e4a"}, + {file = "multidict-6.0.5-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:fe5d7785250541f7f5019ab9cba2c71169dc7d74d0f45253f8313f436458a4ef"}, + {file = "multidict-6.0.5-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:c1c1496e73051918fcd4f58ff2e0f2f3066d1c76a0c6aeffd9b45d53243702cc"}, + {file = "multidict-6.0.5-cp310-cp310-win32.whl", hash = "sha256:7afcdd1fc07befad18ec4523a782cde4e93e0a2bf71239894b8d61ee578c1319"}, + {file = "multidict-6.0.5-cp310-cp310-win_amd64.whl", hash = "sha256:99f60d34c048c5c2fabc766108c103612344c46e35d4ed9ae0673d33c8fb26e8"}, + {file = "multidict-6.0.5-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:f285e862d2f153a70586579c15c44656f888806ed0e5b56b64489afe4a2dbfba"}, + {file = "multidict-6.0.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:53689bb4e102200a4fafa9de9c7c3c212ab40a7ab2c8e474491914d2305f187e"}, + {file = "multidict-6.0.5-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:612d1156111ae11d14afaf3a0669ebf6c170dbb735e510a7438ffe2369a847fd"}, + {file = "multidict-6.0.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7be7047bd08accdb7487737631d25735c9a04327911de89ff1b26b81745bd4e3"}, + {file = "multidict-6.0.5-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:de170c7b4fe6859beb8926e84f7d7d6c693dfe8e27372ce3b76f01c46e489fcf"}, + {file = "multidict-6.0.5-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:04bde7a7b3de05732a4eb39c94574db1ec99abb56162d6c520ad26f83267de29"}, + {file = "multidict-6.0.5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:85f67aed7bb647f93e7520633d8f51d3cbc6ab96957c71272b286b2f30dc70ed"}, + {file = "multidict-6.0.5-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:425bf820055005bfc8aa9a0b99ccb52cc2f4070153e34b701acc98d201693733"}, + {file = "multidict-6.0.5-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:d3eb1ceec286eba8220c26f3b0096cf189aea7057b6e7b7a2e60ed36b373b77f"}, + {file = "multidict-6.0.5-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:7901c05ead4b3fb75113fb1dd33eb1253c6d3ee37ce93305acd9d38e0b5f21a4"}, + {file = "multidict-6.0.5-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:e0e79d91e71b9867c73323a3444724d496c037e578a0e1755ae159ba14f4f3d1"}, + {file = "multidict-6.0.5-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:29bfeb0dff5cb5fdab2023a7a9947b3b4af63e9c47cae2a10ad58394b517fddc"}, + {file = "multidict-6.0.5-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:e030047e85cbcedbfc073f71836d62dd5dadfbe7531cae27789ff66bc551bd5e"}, + {file = "multidict-6.0.5-cp311-cp311-win32.whl", hash = "sha256:2f4848aa3baa109e6ab81fe2006c77ed4d3cd1e0ac2c1fbddb7b1277c168788c"}, + {file = "multidict-6.0.5-cp311-cp311-win_amd64.whl", hash = "sha256:2faa5ae9376faba05f630d7e5e6be05be22913782b927b19d12b8145968a85ea"}, + {file = "multidict-6.0.5-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:51d035609b86722963404f711db441cf7134f1889107fb171a970c9701f92e1e"}, + {file = "multidict-6.0.5-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:cbebcd5bcaf1eaf302617c114aa67569dd3f090dd0ce8ba9e35e9985b41ac35b"}, + {file = "multidict-6.0.5-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:2ffc42c922dbfddb4a4c3b438eb056828719f07608af27d163191cb3e3aa6cc5"}, + {file = "multidict-6.0.5-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ceb3b7e6a0135e092de86110c5a74e46bda4bd4fbfeeb3a3bcec79c0f861e450"}, + {file = "multidict-6.0.5-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:79660376075cfd4b2c80f295528aa6beb2058fd289f4c9252f986751a4cd0496"}, + {file = "multidict-6.0.5-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e4428b29611e989719874670fd152b6625500ad6c686d464e99f5aaeeaca175a"}, + {file = "multidict-6.0.5-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d84a5c3a5f7ce6db1f999fb9438f686bc2e09d38143f2d93d8406ed2dd6b9226"}, + {file = "multidict-6.0.5-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:76c0de87358b192de7ea9649beb392f107dcad9ad27276324c24c91774ca5271"}, + {file = "multidict-6.0.5-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:79a6d2ba910adb2cbafc95dad936f8b9386e77c84c35bc0add315b856d7c3abb"}, + {file = "multidict-6.0.5-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:92d16a3e275e38293623ebf639c471d3e03bb20b8ebb845237e0d3664914caef"}, + {file = "multidict-6.0.5-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:fb616be3538599e797a2017cccca78e354c767165e8858ab5116813146041a24"}, + {file = "multidict-6.0.5-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:14c2976aa9038c2629efa2c148022ed5eb4cb939e15ec7aace7ca932f48f9ba6"}, + {file = "multidict-6.0.5-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:435a0984199d81ca178b9ae2c26ec3d49692d20ee29bc4c11a2a8d4514c67eda"}, + {file = "multidict-6.0.5-cp312-cp312-win32.whl", hash = "sha256:9fe7b0653ba3d9d65cbe7698cca585bf0f8c83dbbcc710db9c90f478e175f2d5"}, + {file = "multidict-6.0.5-cp312-cp312-win_amd64.whl", hash = "sha256:01265f5e40f5a17f8241d52656ed27192be03bfa8764d88e8220141d1e4b3556"}, + {file = "multidict-6.0.5-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:19fe01cea168585ba0f678cad6f58133db2aa14eccaf22f88e4a6dccadfad8b3"}, + {file = "multidict-6.0.5-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6bf7a982604375a8d49b6cc1b781c1747f243d91b81035a9b43a2126c04766f5"}, + {file = "multidict-6.0.5-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:107c0cdefe028703fb5dafe640a409cb146d44a6ae201e55b35a4af8e95457dd"}, + {file = "multidict-6.0.5-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:403c0911cd5d5791605808b942c88a8155c2592e05332d2bf78f18697a5fa15e"}, + {file = "multidict-6.0.5-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:aeaf541ddbad8311a87dd695ed9642401131ea39ad7bc8cf3ef3967fd093b626"}, + {file = "multidict-6.0.5-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e4972624066095e52b569e02b5ca97dbd7a7ddd4294bf4e7247d52635630dd83"}, + {file = "multidict-6.0.5-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:d946b0a9eb8aaa590df1fe082cee553ceab173e6cb5b03239716338629c50c7a"}, + {file = "multidict-6.0.5-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:b55358304d7a73d7bdf5de62494aaf70bd33015831ffd98bc498b433dfe5b10c"}, + {file = "multidict-6.0.5-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:a3145cb08d8625b2d3fee1b2d596a8766352979c9bffe5d7833e0503d0f0b5e5"}, + {file = "multidict-6.0.5-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:d65f25da8e248202bd47445cec78e0025c0fe7582b23ec69c3b27a640dd7a8e3"}, + {file = "multidict-6.0.5-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:c9bf56195c6bbd293340ea82eafd0071cb3d450c703d2c93afb89f93b8386ccc"}, + {file = "multidict-6.0.5-cp37-cp37m-win32.whl", hash = "sha256:69db76c09796b313331bb7048229e3bee7928eb62bab5e071e9f7fcc4879caee"}, + {file = "multidict-6.0.5-cp37-cp37m-win_amd64.whl", hash = "sha256:fce28b3c8a81b6b36dfac9feb1de115bab619b3c13905b419ec71d03a3fc1423"}, + {file = "multidict-6.0.5-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:76f067f5121dcecf0d63a67f29080b26c43c71a98b10c701b0677e4a065fbd54"}, + {file = "multidict-6.0.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:b82cc8ace10ab5bd93235dfaab2021c70637005e1ac787031f4d1da63d493c1d"}, + {file = "multidict-6.0.5-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:5cb241881eefd96b46f89b1a056187ea8e9ba14ab88ba632e68d7a2ecb7aadf7"}, + {file = "multidict-6.0.5-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e8e94e6912639a02ce173341ff62cc1201232ab86b8a8fcc05572741a5dc7d93"}, + {file = "multidict-6.0.5-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:09a892e4a9fb47331da06948690ae38eaa2426de97b4ccbfafbdcbe5c8f37ff8"}, + {file = "multidict-6.0.5-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:55205d03e8a598cfc688c71ca8ea5f66447164efff8869517f175ea632c7cb7b"}, + {file = "multidict-6.0.5-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:37b15024f864916b4951adb95d3a80c9431299080341ab9544ed148091b53f50"}, + {file = "multidict-6.0.5-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f2a1dee728b52b33eebff5072817176c172050d44d67befd681609b4746e1c2e"}, + {file = "multidict-6.0.5-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:edd08e6f2f1a390bf137080507e44ccc086353c8e98c657e666c017718561b89"}, + {file = "multidict-6.0.5-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:60d698e8179a42ec85172d12f50b1668254628425a6bd611aba022257cac1386"}, + {file = "multidict-6.0.5-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:3d25f19500588cbc47dc19081d78131c32637c25804df8414463ec908631e453"}, + {file = "multidict-6.0.5-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:4cc0ef8b962ac7a5e62b9e826bd0cd5040e7d401bc45a6835910ed699037a461"}, + {file = "multidict-6.0.5-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:eca2e9d0cc5a889850e9bbd68e98314ada174ff6ccd1129500103df7a94a7a44"}, + {file = "multidict-6.0.5-cp38-cp38-win32.whl", hash = "sha256:4a6a4f196f08c58c59e0b8ef8ec441d12aee4125a7d4f4fef000ccb22f8d7241"}, + {file = "multidict-6.0.5-cp38-cp38-win_amd64.whl", hash = "sha256:0275e35209c27a3f7951e1ce7aaf93ce0d163b28948444bec61dd7badc6d3f8c"}, + {file = "multidict-6.0.5-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:e7be68734bd8c9a513f2b0cfd508802d6609da068f40dc57d4e3494cefc92929"}, + {file = "multidict-6.0.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:1d9ea7a7e779d7a3561aade7d596649fbecfa5c08a7674b11b423783217933f9"}, + {file = "multidict-6.0.5-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ea1456df2a27c73ce51120fa2f519f1bea2f4a03a917f4a43c8707cf4cbbae1a"}, + {file = "multidict-6.0.5-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:cf590b134eb70629e350691ecca88eac3e3b8b3c86992042fb82e3cb1830d5e1"}, + {file = "multidict-6.0.5-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5c0631926c4f58e9a5ccce555ad7747d9a9f8b10619621f22f9635f069f6233e"}, + {file = "multidict-6.0.5-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:dce1c6912ab9ff5f179eaf6efe7365c1f425ed690b03341911bf4939ef2f3046"}, + {file = "multidict-6.0.5-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c0868d64af83169e4d4152ec612637a543f7a336e4a307b119e98042e852ad9c"}, + {file = "multidict-6.0.5-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:141b43360bfd3bdd75f15ed811850763555a251e38b2405967f8e25fb43f7d40"}, + {file = "multidict-6.0.5-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:7df704ca8cf4a073334e0427ae2345323613e4df18cc224f647f251e5e75a527"}, + {file = "multidict-6.0.5-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:6214c5a5571802c33f80e6c84713b2c79e024995b9c5897f794b43e714daeec9"}, + {file = "multidict-6.0.5-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:cd6c8fca38178e12c00418de737aef1261576bd1b6e8c6134d3e729a4e858b38"}, + {file = "multidict-6.0.5-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:e02021f87a5b6932fa6ce916ca004c4d441509d33bbdbeca70d05dff5e9d2479"}, + {file = "multidict-6.0.5-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ebd8d160f91a764652d3e51ce0d2956b38efe37c9231cd82cfc0bed2e40b581c"}, + {file = "multidict-6.0.5-cp39-cp39-win32.whl", hash = "sha256:04da1bb8c8dbadf2a18a452639771951c662c5ad03aefe4884775454be322c9b"}, + {file = "multidict-6.0.5-cp39-cp39-win_amd64.whl", hash = "sha256:d6f6d4f185481c9669b9447bf9d9cf3b95a0e9df9d169bbc17e363b7d5487755"}, + {file = "multidict-6.0.5-py3-none-any.whl", hash = "sha256:0d63c74e3d7ab26de115c49bffc92cc77ed23395303d496eae515d4204a625e7"}, + {file = "multidict-6.0.5.tar.gz", hash = "sha256:f7e301075edaf50500f0b341543c41194d8df3ae5caf4702f2095f3ca73dd8da"}, +] + +[[package]] +name = "mypy-extensions" +version = "1.0.0" +description = "Type system extensions for programs checked with the mypy type checker." +optional = false +python-versions = ">=3.5" +files = [ + {file = "mypy_extensions-1.0.0-py3-none-any.whl", hash = "sha256:4392f6c0eb8a5668a69e23d168ffa70f0be9ccfd32b5cc2d26a34ae5b844552d"}, + {file = "mypy_extensions-1.0.0.tar.gz", hash = "sha256:75dbf8955dc00442a438fc4d0666508a9a97b6bd41aa2f0ffe9d2f2725af0782"}, +] + +[[package]] +name = "packaging" +version = "23.2" +description = "Core utilities for Python packages" +optional = false +python-versions = ">=3.7" +files = [ + {file = "packaging-23.2-py3-none-any.whl", hash = "sha256:8c491190033a9af7e1d931d0b5dacc2ef47509b34dd0de67ed209b5203fc88c7"}, + {file = "packaging-23.2.tar.gz", hash = "sha256:048fb0e9405036518eaaf48a55953c750c11e1a1b68e0dd1a9d62ed0c092cfc5"}, +] + +[[package]] +name = "pathspec" +version = "0.12.1" +description = "Utility library for gitignore style pattern matching of file paths." +optional = false +python-versions = ">=3.8" +files = [ + {file = "pathspec-0.12.1-py3-none-any.whl", hash = "sha256:a0d503e138a4c123b27490a4f7beda6a01c6f288df0e4a8b79c7eb0dc7b4cc08"}, + {file = "pathspec-0.12.1.tar.gz", hash = "sha256:a482d51503a1ab33b1c67a6c3813a26953dbdc71c31dacaef9a838c4e29f5712"}, +] + +[[package]] +name = "pip" +version = "24.0" +description = "The PyPA recommended tool for installing Python packages." +optional = false +python-versions = ">=3.7" +files = [ + {file = "pip-24.0-py3-none-any.whl", hash = "sha256:ba0d021a166865d2265246961bec0152ff124de910c5cc39f1156ce3fa7c69dc"}, + {file = "pip-24.0.tar.gz", hash = "sha256:ea9bd1a847e8c5774a5777bb398c19e80bcd4e2aa16a4b301b718fe6f593aba2"}, +] + +[[package]] +name = "platformdirs" +version = "4.2.0" +description = "A small Python package for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." +optional = false +python-versions = ">=3.8" +files = [ + {file = "platformdirs-4.2.0-py3-none-any.whl", hash = "sha256:0614df2a2f37e1a662acbd8e2b25b92ccf8632929bc6d43467e17fe89c75e068"}, + {file = "platformdirs-4.2.0.tar.gz", hash = "sha256:ef0cc731df711022c174543cb70a9b5bd22e5a9337c8624ef2c2ceb8ddad8768"}, +] + +[package.extras] +docs = ["furo (>=2023.9.10)", "proselint (>=0.13)", "sphinx (>=7.2.6)", "sphinx-autodoc-typehints (>=1.25.2)"] +test = ["appdirs (==1.4.4)", "covdefaults (>=2.3)", "pytest (>=7.4.3)", "pytest-cov (>=4.1)", "pytest-mock (>=3.12)"] + +[[package]] +name = "pluggy" +version = "1.4.0" +description = "plugin and hook calling mechanisms for python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pluggy-1.4.0-py3-none-any.whl", hash = "sha256:7db9f7b503d67d1c5b95f59773ebb58a8c1c288129a88665838012cfb07b8981"}, + {file = "pluggy-1.4.0.tar.gz", hash = "sha256:8c85c2876142a764e5b7548e7d9a0e0ddb46f5185161049a79b7e974454223be"}, +] + +[package.extras] +dev = ["pre-commit", "tox"] +testing = ["pytest", "pytest-benchmark"] + +[[package]] +name = "pycodestyle" +version = "2.11.1" +description = "Python style guide checker" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pycodestyle-2.11.1-py2.py3-none-any.whl", hash = "sha256:44fe31000b2d866f2e41841b18528a505fbd7fef9017b04eff4e2648a0fadc67"}, + {file = "pycodestyle-2.11.1.tar.gz", hash = "sha256:41ba0e7afc9752dfb53ced5489e89f8186be00e599e712660695b7a75ff2663f"}, +] + +[[package]] +name = "pyflakes" +version = "3.1.0" +description = "passive checker of Python programs" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pyflakes-3.1.0-py2.py3-none-any.whl", hash = "sha256:4132f6d49cb4dae6819e5379898f2b8cce3c5f23994194c24b77d5da2e36f774"}, + {file = "pyflakes-3.1.0.tar.gz", hash = "sha256:a0aae034c444db0071aa077972ba4768d40c830d9539fd45bf4cd3f8f6992efc"}, +] + +[[package]] +name = "pygments" +version = "2.17.2" +description = "Pygments is a syntax highlighting package written in Python." +optional = false +python-versions = ">=3.7" +files = [ + {file = "pygments-2.17.2-py3-none-any.whl", hash = "sha256:b27c2826c47d0f3219f29554824c30c5e8945175d888647acd804ddd04af846c"}, + {file = "pygments-2.17.2.tar.gz", hash = "sha256:da46cec9fd2de5be3a8a784f434e4c4ab670b4ff54d605c4c2717e9d49c4c367"}, +] + +[package.extras] +plugins = ["importlib-metadata"] +windows-terminal = ["colorama (>=0.4.6)"] + +[[package]] +name = "pylint" +version = "3.0.3" +description = "python code static checker" +optional = false +python-versions = ">=3.8.0" +files = [ + {file = "pylint-3.0.3-py3-none-any.whl", hash = "sha256:7a1585285aefc5165db81083c3e06363a27448f6b467b3b0f30dbd0ac1f73810"}, + {file = "pylint-3.0.3.tar.gz", hash = "sha256:58c2398b0301e049609a8429789ec6edf3aabe9b6c5fec916acd18639c16de8b"}, +] + +[package.dependencies] +astroid = ">=3.0.1,<=3.1.0-dev0" +colorama = {version = ">=0.4.5", markers = "sys_platform == \"win32\""} +dill = [ + {version = ">=0.3.7", markers = "python_version >= \"3.12\""}, + {version = ">=0.3.6", markers = "python_version >= \"3.11\" and python_version < \"3.12\""}, +] +isort = ">=4.2.5,<5.13.0 || >5.13.0,<6" +mccabe = ">=0.6,<0.8" +platformdirs = ">=2.2.0" +tomlkit = ">=0.10.1" + +[package.extras] +spelling = ["pyenchant (>=3.2,<4.0)"] +testutils = ["gitpython (>3)"] + +[[package]] +name = "pysocks" +version = "1.7.1" +description = "A Python SOCKS client module. See https://github.com/Anorov/PySocks for more information." +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +files = [ + {file = "PySocks-1.7.1-py27-none-any.whl", hash = "sha256:08e69f092cc6dbe92a0fdd16eeb9b9ffbc13cadfe5ca4c7bd92ffb078b293299"}, + {file = "PySocks-1.7.1-py3-none-any.whl", hash = "sha256:2725bd0a9925919b9b51739eea5f9e2bae91e83288108a9ad338b2e3a4435ee5"}, + {file = "PySocks-1.7.1.tar.gz", hash = "sha256:3f8804571ebe159c380ac6de37643bb4685970655d3bba243530d6558b799aa0"}, +] + +[[package]] +name = "pytest" +version = "7.4.4" +description = "pytest: simple powerful testing with Python" +optional = false +python-versions = ">=3.7" +files = [ + {file = "pytest-7.4.4-py3-none-any.whl", hash = "sha256:b090cdf5ed60bf4c45261be03239c2c1c22df034fbffe691abe93cd80cea01d8"}, + {file = "pytest-7.4.4.tar.gz", hash = "sha256:2cf0005922c6ace4a3e2ec8b4080eb0d9753fdc93107415332f50ce9e7994280"}, +] + +[package.dependencies] +colorama = {version = "*", markers = "sys_platform == \"win32\""} +iniconfig = "*" +packaging = "*" +pluggy = ">=0.12,<2.0" + +[package.extras] +testing = ["argcomplete", "attrs (>=19.2.0)", "hypothesis (>=3.56)", "mock", "nose", "pygments (>=2.7.2)", "requests", "setuptools", "xmlschema"] + +[[package]] +name = "pytest-cov" +version = "4.1.0" +description = "Pytest plugin for measuring coverage." +optional = false +python-versions = ">=3.7" +files = [ + {file = "pytest-cov-4.1.0.tar.gz", hash = "sha256:3904b13dfbfec47f003b8e77fd5b589cd11904a21ddf1ab38a64f204d6a10ef6"}, + {file = "pytest_cov-4.1.0-py3-none-any.whl", hash = "sha256:6ba70b9e97e69fcc3fb45bfeab2d0a138fb65c4d0d6a41ef33983ad114be8c3a"}, +] + +[package.dependencies] +coverage = {version = ">=5.2.1", extras = ["toml"]} +pytest = ">=4.6" + +[package.extras] +testing = ["fields", "hunter", "process-tests", "pytest-xdist", "six", "virtualenv"] + +[[package]] +name = "pytest-pspec" +version = "0.0.4" +description = "A rspec format reporter for Python ptest" +optional = false +python-versions = "*" +files = [ + {file = "pytest-pspec-0.0.4.tar.gz", hash = "sha256:5c0b0c9e964d5066cc5f2a2e1b296ad1313abbf58c4fd75014553fdf65bfe67a"}, + {file = "pytest_pspec-0.0.4-py2.py3-none-any.whl", hash = "sha256:f80cc46f8896524bfe68750f3a5324bad8d4cb5112e90004bfdaafb7248e5cfd"}, +] + +[package.dependencies] +pytest = ">=3.0.0" +six = ">=1.11.0" + +[[package]] +name = "python-dateutil" +version = "2.8.2" +description = "Extensions to the standard Python datetime module" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7" +files = [ + {file = "python-dateutil-2.8.2.tar.gz", hash = "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86"}, + {file = "python_dateutil-2.8.2-py2.py3-none-any.whl", hash = "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9"}, +] + +[package.dependencies] +six = ">=1.5" + +[[package]] +name = "python-dotenv" +version = "1.0.1" +description = "Read key-value pairs from a .env file and set them as environment variables" +optional = false +python-versions = ">=3.8" +files = [ + {file = "python-dotenv-1.0.1.tar.gz", hash = "sha256:e324ee90a023d808f1959c46bcbc04446a10ced277783dc6ee09987c37ec10ca"}, + {file = "python_dotenv-1.0.1-py3-none-any.whl", hash = "sha256:f7b63ef50f1b690dddf550d03497b66d609393b40b564ed0d674909a68ebf16a"}, +] + +[package.extras] +cli = ["click (>=5.0)"] + +[[package]] +name = "redis" +version = "4.6.0" +description = "Python client for Redis database and key-value store" +optional = false +python-versions = ">=3.7" +files = [ + {file = "redis-4.6.0-py3-none-any.whl", hash = "sha256:e2b03db868160ee4591de3cb90d40ebb50a90dd302138775937f6a42b7ed183c"}, + {file = "redis-4.6.0.tar.gz", hash = "sha256:585dc516b9eb042a619ef0a39c3d7d55fe81bdb4df09a52c9cdde0d07bf1aa7d"}, +] + +[package.dependencies] +async-timeout = {version = ">=4.0.2", markers = "python_full_version <= \"3.11.2\""} + +[package.extras] +hiredis = ["hiredis (>=1.0.0)"] +ocsp = ["cryptography (>=36.0.1)", "pyopenssl (==20.0.1)", "requests (>=2.26.0)"] + +[[package]] +name = "requests" +version = "2.31.0" +description = "Python HTTP for Humans." +optional = false +python-versions = ">=3.7" +files = [ + {file = "requests-2.31.0-py3-none-any.whl", hash = "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f"}, + {file = "requests-2.31.0.tar.gz", hash = "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1"}, +] + +[package.dependencies] +certifi = ">=2017.4.17" +charset-normalizer = ">=2,<4" +idna = ">=2.5,<4" +PySocks = {version = ">=1.5.6,<1.5.7 || >1.5.7", optional = true, markers = "extra == \"socks\""} +urllib3 = ">=1.21.1,<3" + +[package.extras] +socks = ["PySocks (>=1.5.6,!=1.5.7)"] +use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] + +[[package]] +name = "requests-toolbelt" +version = "1.0.0" +description = "A utility belt for advanced users of python-requests" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +files = [ + {file = "requests-toolbelt-1.0.0.tar.gz", hash = "sha256:7681a0a3d047012b5bdc0ee37d7f8f07ebe76ab08caeccfc3921ce23c88d5bc6"}, + {file = "requests_toolbelt-1.0.0-py2.py3-none-any.whl", hash = "sha256:cccfdd665f0a24fcf4726e690f65639d272bb0637b9b92dfd91a5568ccf6bd06"}, +] + +[package.dependencies] +requests = ">=2.0.1,<3.0.0" + +[[package]] +name = "rich" +version = "13.7.0" +description = "Render rich text, tables, progress bars, syntax highlighting, markdown and more to the terminal" +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "rich-13.7.0-py3-none-any.whl", hash = "sha256:6da14c108c4866ee9520bbffa71f6fe3962e193b7da68720583850cd4548e235"}, + {file = "rich-13.7.0.tar.gz", hash = "sha256:5cb5123b5cf9ee70584244246816e9114227e0b98ad9176eede6ad54bf5403fa"}, +] + +[package.dependencies] +markdown-it-py = ">=2.2.0" +pygments = ">=2.13.0,<3.0.0" + +[package.extras] +jupyter = ["ipywidgets (>=7.5.1,<9)"] + +[[package]] +name = "setuptools" +version = "69.0.3" +description = "Easily download, build, install, upgrade, and uninstall Python packages" +optional = false +python-versions = ">=3.8" +files = [ + {file = "setuptools-69.0.3-py3-none-any.whl", hash = "sha256:385eb4edd9c9d5c17540511303e39a147ce2fc04bc55289c322b9e5904fe2c05"}, + {file = "setuptools-69.0.3.tar.gz", hash = "sha256:be1af57fc409f93647f2e8e4573a142ed38724b8cdd389706a867bb4efcf1e78"}, +] + +[package.extras] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"] +testing = ["build[virtualenv]", "filelock (>=3.4.0)", "flake8-2020", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pip (>=19.1)", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-ruff", "pytest-timeout", "pytest-xdist", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] +testing-integration = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging (>=23.1)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] + +[[package]] +name = "six" +version = "1.16.0" +description = "Python 2 and 3 compatibility utilities" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" +files = [ + {file = "six-1.16.0-py2.py3-none-any.whl", hash = "sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254"}, + {file = "six-1.16.0.tar.gz", hash = "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926"}, +] + +[[package]] +name = "tomlkit" +version = "0.12.3" +description = "Style preserving TOML library" +optional = false +python-versions = ">=3.7" +files = [ + {file = "tomlkit-0.12.3-py3-none-any.whl", hash = "sha256:b0a645a9156dc7cb5d3a1f0d4bab66db287fcb8e0430bdd4664a095ea16414ba"}, + {file = "tomlkit-0.12.3.tar.gz", hash = "sha256:75baf5012d06501f07bee5bf8e801b9f343e7aac5a92581f20f80ce632e6b5a4"}, +] + +[[package]] +name = "urllib3" +version = "2.2.0" +description = "HTTP library with thread-safe connection pooling, file post, and more." +optional = false +python-versions = ">=3.8" +files = [ + {file = "urllib3-2.2.0-py3-none-any.whl", hash = "sha256:ce3711610ddce217e6d113a2732fafad960a03fd0318c91faa79481e35c11224"}, + {file = "urllib3-2.2.0.tar.gz", hash = "sha256:051d961ad0c62a94e50ecf1af379c3aba230c66c710493493560c0c223c49f20"}, +] + +[package.extras] +brotli = ["brotli (>=1.0.9)", "brotlicffi (>=0.8.0)"] +h2 = ["h2 (>=4,<5)"] +socks = ["pysocks (>=1.5.6,!=1.5.7,<2.0)"] +zstd = ["zstandard (>=0.18.0)"] + +[[package]] +name = "werkzeug" +version = "3.0.1" +description = "The comprehensive WSGI web application library." +optional = false +python-versions = ">=3.8" +files = [ + {file = "werkzeug-3.0.1-py3-none-any.whl", hash = "sha256:90a285dc0e42ad56b34e696398b8122ee4c681833fb35b8334a095d82c56da10"}, + {file = "werkzeug-3.0.1.tar.gz", hash = "sha256:507e811ecea72b18a404947aded4b3390e1db8f826b494d76550ef45bb3b1dcc"}, +] + +[package.dependencies] +MarkupSafe = ">=2.1.1" + +[package.extras] +watchdog = ["watchdog (>=2.3)"] + +[metadata] +lock-version = "2.0" +python-versions = "^3.11" +content-hash = "62db235e00d94e8e52c69d7ccb07b7c1e41765156692349e33eb6c4ee36b432f" diff --git a/pyproject.toml b/pyproject.toml index 89d82a6..3dfe53d 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -2,16 +2,16 @@ name = "lab-flask-rest" version = "0.1.0" description = "REST API with Flask Lab" -authors = ["Your Name "] +authors = ["John Rofrano"] license = "Apache 2.0" readme = "README.md" [tool.poetry.dependencies] -python = "^3.9" -Flask = "^2.1.2" +python = "^3.11" +Flask = "^3.0.2" redis = "^4.5.3" python-dotenv = "^1.0.0" -gunicorn = "^20.1.0" +gunicorn = "^21.2.0" honcho = "^1.1.0" [tool.poetry.group.dev.dependencies] @@ -23,9 +23,47 @@ pytest-pspec = "^0.0.4" pytest-cov = "^4.1.0" factory-boy = "^3.3.0" coverage = "^7.3.2" -green = "^3.4.3" httpie = "^3.2.2" [build-system] requires = ["poetry-core"] build-backend = "poetry.core.masonry.api" + +# +# Tool configurations +# + +[tool.pylint.'MESSAGES CONTROL'] +max-line-length = 127 +disable = "no-member,protected-access,global-statement" + +[tool.pytest.ini_options] +minversion = "6.0" +addopts = "--pspec --cov=service --cov-fail-under=95" +testpaths = ["tests"] + +[tool.coverage.run] +source = ["service"] +omit = [ + "venv/*", + ".venv/*" +] + +[tool.coverage.report] +show_missing = true +exclude_lines = [ + "pragma: no cover", + "pragma: no branch", + "pass", + "subprocess.CalledProcessError", + "sys.exit", + "if __name__ == .__main__.:" +] +ignore_errors = true + +[tool.coverage.xml] +output="./coverage.xml" + +[tool.coverage.html] +title = "Test Coverage Report" +directory = "coverage_html_report" diff --git a/requirements.txt b/requirements.txt deleted file mode 100644 index a4735dc..0000000 --- a/requirements.txt +++ /dev/null @@ -1,24 +0,0 @@ -# Werkzeug keeps breaking Flask! -Werkzeug==2.1.2 - -# Runtime dependencies -Flask==2.1.2 -redis==4.5.3 -python-dotenv==0.20.0 - -# Runtime tools -gunicorn==20.1.0 -honcho==1.1.0 - -# Code quality -pylint==2.14.0 -flake8==4.0.1 -black==22.3.0 - -# Testing dependencies -nose==1.3.7 -pinocchio==0.4.3 -coverage==6.3.2 - -# Utilities -httpie==3.2.1 diff --git a/setup.cfg b/setup.cfg index f41e5e5..ca0dd4b 100644 --- a/setup.cfg +++ b/setup.cfg @@ -1,18 +1,6 @@ -[nosetests] -verbosity=2 -with-spec=1 -spec-color=1 -with-coverage=1 -cover-erase=1 -cover-package=service -cover-xml=1 -cover-xml-file=./coverage.xml -# with-xunit=1 -# xunit-file=./unittests.xml - -[coverage:report] -show_missing = True +# Code Quality [flake8] +max-line-length = 127 per-file-ignores = - */__init__.py: E402 + */__init__.py: F401 E402 From 8a8635fb8ce46beb310580fa631c749151ec3bc9 Mon Sep 17 00:00:00 2001 From: John Rofrano Date: Sun, 4 Feb 2024 16:02:59 +0000 Subject: [PATCH 2/5] Fixed linting errors --- service/routes.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/service/routes.py b/service/routes.py index f32d055..e5bba0a 100644 --- a/service/routes.py +++ b/service/routes.py @@ -78,7 +78,7 @@ def create_counters(name): Returns: dict: the counter and it's value """ - app.logger.info(f"Request to Create counter {name}...") + app.logger.info("Request to Create counter %s...", name) # See if the counter already exists and send an error if it does counter = Counter.find(name) @@ -111,7 +111,7 @@ def read_counters(name): Returns: dict: the counter and it's value """ - app.logger.info(f"Request to Read counter {name}...") + app.logger.info("Request to Read counter %s...", name) # Get the current counter counter = Counter.find(name) @@ -135,7 +135,7 @@ def update_counters(name): Returns: dict: the counter and it's value """ - app.logger.info(f"Request to Update counter {name}...") + app.logger.info("Request to Update counter %s...", name) # Get the current counter counter = Counter.find(name) @@ -161,7 +161,7 @@ def delete_counters(name): Returns: str: always returns an empty string """ - app.logger.info(f"Request to Delete counter {name}...") + app.logger.info("Request to Delete counter %s...", name) # Get the current counter counter = Counter.find(name) @@ -186,7 +186,7 @@ def reset_counters(name): Returns: dict: the counter and it's zero value """ - app.logger.info(f"Request to Reset counter {name}...") + app.logger.info("Request to Reset counter %s...", name) # Get the current counter counter = Counter.find(name) From c245dc42a1e7341e4b62dc074a7f06d4acecdb71 Mon Sep 17 00:00:00 2001 From: John Rofrano Date: Sun, 4 Feb 2024 16:30:29 +0000 Subject: [PATCH 3/5] Add app factory --- Procfile | 2 +- poetry.lock | 21 ++- pyproject.toml | 1 + service/__init__.py | 90 ++++++++----- service/common/error_handlers.py | 61 +++------ service/common/status.py | 1 + service/config.py | 9 +- service/models.py | 198 +++++++++++++++++----------- service/routes.py | 193 ++++++++++----------------- tests/test_models.py | 218 ++++++++++++++++++------------- tests/test_routes.py | 212 ++++++++++++++++++------------ wsgi.py | 9 ++ 12 files changed, 555 insertions(+), 460 deletions(-) create mode 100644 wsgi.py diff --git a/Procfile b/Procfile index 8ec5faf..c1ee86f 100644 --- a/Procfile +++ b/Procfile @@ -1 +1 @@ -web: gunicorn --bind 0.0.0.0:$PORT --log-level=info service:app +web: gunicorn --bind 0.0.0.0:$PORT --log-level=info wsgi:app diff --git a/poetry.lock b/poetry.lock index 2b5a210..52a0284 100644 --- a/poetry.lock +++ b/poetry.lock @@ -372,6 +372,25 @@ Werkzeug = ">=3.0.0" async = ["asgiref (>=3.2)"] dotenv = ["python-dotenv"] +[[package]] +name = "flask-redis" +version = "0.4.0" +description = "A nice way to use Redis in your Flask app" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +files = [ + {file = "flask-redis-0.4.0.tar.gz", hash = "sha256:e1fccc11e7ea35c2a4d68c0b9aa58226a098e45e834d615c7b6c4928b01ddd6c"}, + {file = "flask_redis-0.4.0-py2.py3-none-any.whl", hash = "sha256:8d79eef4eb1217095edab603acc52f935b983ae4b7655ee7c82c0dfd87315d17"}, +] + +[package.dependencies] +Flask = ">=0.8" +redis = ">=2.7.6" + +[package.extras] +dev = ["coverage", "pre-commit", "pytest", "pytest-mock"] +tests = ["coverage", "pytest", "pytest-mock"] + [[package]] name = "gunicorn" version = "21.2.0" @@ -1092,4 +1111,4 @@ watchdog = ["watchdog (>=2.3)"] [metadata] lock-version = "2.0" python-versions = "^3.11" -content-hash = "62db235e00d94e8e52c69d7ccb07b7c1e41765156692349e33eb6c4ee36b432f" +content-hash = "cf73b89a8440ba253119d096b18749a41f7f339ce2a7c0d19c9336ba73ca365d" diff --git a/pyproject.toml b/pyproject.toml index 3dfe53d..7de6c40 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -10,6 +10,7 @@ readme = "README.md" python = "^3.11" Flask = "^3.0.2" redis = "^4.5.3" +flask-redis = "^0.4.0" python-dotenv = "^1.0.0" gunicorn = "^21.2.0" honcho = "^1.1.0" diff --git a/service/__init__.py b/service/__init__.py index d2b6cce..8371cbb 100644 --- a/service/__init__.py +++ b/service/__init__.py @@ -1,38 +1,62 @@ +# Copyright 2016, 2023 John J. Rofrano. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + """ -Package: service Package for the application models and service routes -This module creates and configures the Flask app and sets up the logging -and SQL database """ -import sys from flask import Flask +from flask_redis import FlaskRedis from service import config -from .common import log_handlers - -# Create Flask application -app = Flask(__name__) -app.config.from_object(config) - -# Dependencies require we import the routes AFTER the Flask app is created -# pylint: disable=wrong-import-position, wrong-import-order -from service import routes -# flake8: noqa: F401 -# pylint: disable=wrong-import-position -from .common import error_handlers -from .models import Counter - -# Set up logging for production -log_handlers.init_logging(app, "gunicorn.error") - -app.logger.info(70 * "*") -app.logger.info(" S E R V I C E R U N N I N G ".center(70, "*")) -app.logger.info(70 * "*") - -try: - Counter.init_db(app) # Initialize Redis -except Exception as error: # pylint: disable=broad-except - app.logger.critical("%s: Cannot continue", error) - # gunicorn requires exit code 4 to stop spawning workers when they die - sys.exit(4) - -app.logger.info("Service initialized!") +from service.common import log_handlers + +# Globally accessible libraries +# redis = FlaskRedis() + + +############################################################ +# Initialize the Flask instance +############################################################ +def init_app(): + """Initialize the core application.""" + app = Flask(__name__) + app.config.from_object(config) + + # Initialize Plugins + # redis.init_app(app) + + with app.app_context(): + # Include our Routes + + # pylint: disable=import-outside-toplevel, unused-import + from service import routes, models + from service.common import error_handlers + + # Set up logging for production + log_handlers.init_logging(app, "gunicorn.error") + + app.logger.info(70 * "*") + app.logger.info(" H I T C O U N T E R S E R V I C E ".center(70, "*")) + app.logger.info(70 * "*") + + app.logger.info("Service initialized!") + + # Initialize the database + try: + app.logger.info("Initializing the Redis database") + models.Counter.connect(app.config['DATABASE_URI']) + app.logger.info("Connected!") + except models.DatabaseConnectionError as err: + app.logger.error(str(err)) + + return app diff --git a/service/common/error_handlers.py b/service/common/error_handlers.py index 1108697..ce7ed6b 100644 --- a/service/common/error_handlers.py +++ b/service/common/error_handlers.py @@ -15,27 +15,17 @@ ###################################################################### """ -Module: error_handlers +Error Handlers + +This module contains error handlers functions to send back errors as json """ from flask import jsonify -from service import app -from . import status - +from flask import current_app as app +from service.common import status ###################################################################### # Error Handlers ###################################################################### -@app.errorhandler(status.HTTP_400_BAD_REQUEST) -def bad_request(error): - """Handles bad requests with 400_BAD_REQUEST""" - message = str(error) - app.logger.warning(message) - return ( - jsonify( - status=status.HTTP_400_BAD_REQUEST, error="Bad Request", message=message - ), - status.HTTP_400_BAD_REQUEST, - ) @app.errorhandler(status.HTTP_404_NOT_FOUND) @@ -64,46 +54,31 @@ def method_not_supported(error): ) -@app.errorhandler(status.HTTP_409_CONFLICT) -def resource_conflict(error): - """Handles resource conflicts with HTTP_409_CONFLICT""" - message = str(error) - app.logger.warning(message) - return ( - jsonify( - status=status.HTTP_409_CONFLICT, - error="Conflict", - message=message, - ), - status.HTTP_409_CONFLICT, - ) - - -@app.errorhandler(status.HTTP_415_UNSUPPORTED_MEDIA_TYPE) -def mediatype_not_supported(error): - """Handles unsupported media requests with 415_UNSUPPORTED_MEDIA_TYPE""" +@app.errorhandler(status.HTTP_500_INTERNAL_SERVER_ERROR) +def internal_server_error(error): + """Handles unexpected server error with 500_SERVER_ERROR""" message = str(error) - app.logger.warning(message) + app.logger.error(message) return ( jsonify( - status=status.HTTP_415_UNSUPPORTED_MEDIA_TYPE, - error="Unsupported media type", + status=status.HTTP_500_INTERNAL_SERVER_ERROR, + error="Internal Server Error", message=message, ), - status.HTTP_415_UNSUPPORTED_MEDIA_TYPE, + status.HTTP_500_INTERNAL_SERVER_ERROR, ) -@app.errorhandler(status.HTTP_500_INTERNAL_SERVER_ERROR) -def internal_server_error(error): - """Handles unexpected server error with 500_SERVER_ERROR""" +@app.errorhandler(status.HTTP_503_SERVICE_UNAVAILABLE) +def service_unavailable(error): + """Handles unexpected server error with 503_SERVICE_UNAVAILABLE""" message = str(error) app.logger.error(message) return ( jsonify( - status=status.HTTP_500_INTERNAL_SERVER_ERROR, - error="Internal Server Error", + status=status.HTTP_503_SERVICE_UNAVAILABLE, + error="Service is unavailable", message=message, ), - status.HTTP_500_INTERNAL_SERVER_ERROR, + status.HTTP_503_SERVICE_UNAVAILABLE, ) diff --git a/service/common/status.py b/service/common/status.py index 8e6080e..d3d655e 100644 --- a/service/common/status.py +++ b/service/common/status.py @@ -1,3 +1,4 @@ +# coding: utf8 """ Descriptive HTTP status codes, for code readability. See RFC 2616 and RFC 6585. diff --git a/service/config.py b/service/config.py index 1dd122e..e799d56 100644 --- a/service/config.py +++ b/service/config.py @@ -2,9 +2,8 @@ Global Configuration for Application """ import os +import logging -# Get the database from the environment (12 factor) -DATABASE_URI = os.getenv("DATABASE_URI", "redis://localhost:6379") - -# Secret for session management -SECRET_KEY = os.getenv("SECRET_KEY", "s3cr3t-key-shhhh") +# Get configuration from environment +DATABASE_URI = os.getenv("DATABASE_URI", "redis://:@localhost:6379/0") +LOGGING_LEVEL = logging.INFO diff --git a/service/models.py b/service/models.py index 2bbe7b1..8d4b419 100644 --- a/service/models.py +++ b/service/models.py @@ -1,111 +1,163 @@ ###################################################################### -# Copyright (c) 2022 John J. Rofrano. All Rights Reserved. +# Copyright 2016, 2020 John Rofrano. All Rights Reserved. # -# Licensed under the Apache License, Version 2.0 (the "License"); +# Licensed under the Apache License, Version 2.0 (the 'License'); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # https://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an "AS IS" BASIS, +# distributed under the License is distributed on an 'AS IS' BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. ###################################################################### - """ -Models for Counter - -All of the models are stored in this module +Counter Model """ +import os import logging from redis import Redis +from redis.exceptions import ConnectionError as RedisConnectionError + +logger = logging.getLogger(__name__) + +DATABASE_URI = os.getenv("DATABASE_URI", "redis://localhost:6379") -logger = logging.getLogger("flask.app") + +class DatabaseConnectionError(RedisConnectionError): + """Generic Exception for Redis database connection errors""" class Counter: - """ - Class that represents a Counter + """An integer counter that is persisted in Redis + + You can establish a connection to Redis using an environment + variable DATABASE_URI in the following format: + + DATABASE_URI="redis://userid:password@localhost:6379/0" + + This follows the same standards as SQLAlchemy URIs """ - app = None - counter = None + redis = None - def __init__(self, name: str, count: int = 0): - """Constructor for Counter""" + def __init__(self, name: str = "hits", value: int = None): + """Constructor""" self.name = name - self.count = count + if not value: + self.value = 0 + else: + self.value = value - def __repr__(self): - return f"" + @property + def value(self): + """Returns the current value of the counter""" + return int(Counter.redis.get(self.name)) - def create(self): - """ - Creates a Counter to the database - """ - logger.info("Creating %s", self.name) - self.counter.set(self.name, self.count) + @value.setter + def value(self, value): + """Sets the value of the counter""" + Counter.redis.set(self.name, value) - def update(self): - """ - Updates a Counter to the database - """ - logger.info("Saving %s", self.name) - self.count = self.counter.incr(self.name) + @value.deleter + def value(self): + """Removes the counter fom the database""" + Counter.redis.delete(self.name) - def delete(self): - """Removes a Counter from the data store""" - logger.info("Deleting %s", self.name) - self.counter.delete(self.name) - self.count = 0 - - def reset(self): - """ - Resets a Counter to zero - """ - logger.info("Resetting %s", self.name) - self.counter.set(self.name, 0) - self.count = 0 + def increment(self): + """Increments the current value of the counter by 1""" + return Counter.redis.incr(self.name) def serialize(self): - """Serializes a Counter into a dictionary""" - return {"name": self.name, "count": self.count} + """Converts a counter into a dictionary""" + return { + "name": self.name, + "counter": int(Counter.redis.get(self.name)) + } + + ###################################################################### + # F I N D E R M E T H O D S + ###################################################################### @classmethod - def init_db(cls, app): - """Initializes the database session""" - logger.info("Initializing database") - cls.app = app - # This is where we initialize Redis from the Flask app - cls.counter = Redis.from_url( - app.config["DATABASE_URI"], encoding="utf-8", decode_responses=True - ) + def all(cls): + """Returns all of the counters""" + try: + counters = [ + {"name": key, "counter": int(cls.redis.get(key))} + for key in cls.redis.keys("*") + ] + except Exception as err: + raise DatabaseConnectionError(err) from err + return counters @classmethod - def all(cls) -> list: - """Returns all of the Counters in the database""" - logger.info("Processing all Counters") - return [ - Counter(name, int(cls.counter.get(name))) for name in cls.counter.keys("*") - ] + def find(cls, name): + """Finds a counter with the name or returns None""" + counter = None + try: + count = cls.redis.get(name) + if count: + counter = Counter(name, count) + except Exception as err: + raise DatabaseConnectionError(err) from err + return counter @classmethod - def find(cls, name: str): - """Finds a Counter by it's name""" - logger.info("Processing lookup for name %s ...", name) - count = cls.counter.get(name) - if not count: - return None - return Counter(name, int(count)) + def remove_all(cls): + """Removes all of the keys in the database""" + try: + cls.redis.flushall() + except Exception as err: + raise DatabaseConnectionError(err) from err + + ###################################################################### + # R E D I S D A T A B A S E C O N N E C T I O N M E T H O D S + ###################################################################### @classmethod - def remove_all(cls) -> None: - """Removes all counters from the database""" - logger.info("Request to Remove all counters...") - if cls.app.testing: - logger.info("Removing all counters") - cls.counter.flushall() - else: - logger.warning("Cannot Remove all counters, system not under test") + def test_connection(cls): + """Test connection by pinging the host""" + success = False + try: + cls.redis.ping() + logger.info("Connection established") + success = True + except RedisConnectionError: + logger.warning("Connection Error!") + return success + + @classmethod + def connect(cls, database_uri=None): + """Established database connection + + Arguments: + database_uri: a uri to the Redis database + + Raises: + DatabaseConnectionError: Could not connect + """ + if not database_uri: + if "DATABASE_URI" in os.environ and os.environ["DATABASE_URI"]: + database_uri = os.environ["DATABASE_URI"] + else: + msg = "DATABASE_URI is missing from environment." + logger.error(msg) + raise DatabaseConnectionError(msg) + + logger.info("Attempting to connecting to Redis...") + + cls.redis = Redis.from_url( + database_uri, encoding="utf-8", decode_responses=True + ) + + if not cls.test_connection(): + # if you end up here, redis instance is down. + cls.redis = None + logger.fatal("*** FATAL ERROR: Could not connect to the Redis Service") + raise DatabaseConnectionError("Could not connect to the Redis Service") + + logger.info("Successfully connected to Redis") + return cls.redis diff --git a/service/routes.py b/service/routes.py index e5bba0a..e727745 100644 --- a/service/routes.py +++ b/service/routes.py @@ -1,5 +1,5 @@ ###################################################################### -# Copyright (c) 2022 John J. Rofrano. All Rights Reserved. +# Copyright (c) 2015, 2024 John J. Rofrano. All Rights Reserved. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. @@ -19,12 +19,15 @@ This service keeps track of named counters """ -from flask import jsonify, url_for, abort -from service.models import Counter -from .common import status # HTTP Status Codes -# Import Flask application -from . import app +import os +from flask import jsonify, abort, url_for +from flask import current_app as app +from service.common import status # HTTP Status Codes +from .models import Counter, DatabaseConnectionError + +DEBUG = os.getenv("DEBUG", "False") == "True" +PORT = os.getenv("PORT", "8080") ###################################################################### @@ -45,51 +48,56 @@ def index(): ############################################################ -# R E S T A P I -############################################################ - -# ----------------------------------------------------------- # List counters -# ----------------------------------------------------------- +############################################################ @app.route("/counters", methods=["GET"]) def list_counters(): - """Lists all of the counters in the database - - Returns: - list: an array of counter names - """ + """List counters""" app.logger.info("Request to list all counters...") + try: + counters = Counter.all() + except DatabaseConnectionError as err: + abort(status.HTTP_503_SERVICE_UNAVAILABLE, err) - # Get the names of all of the counters - counters = [counter.serialize() for counter in Counter.all()] return jsonify(counters) -# ----------------------------------------------------------- -# Create counters -# ----------------------------------------------------------- -@app.route("/counters/", methods=["POST"]) -def create_counters(name): - """Creates a new counter and stores it in the database +############################################################ +# Read counters +############################################################ +@app.route("/counters/", methods=["GET"]) +def read_counters(name): + """Read a counter""" + app.logger.info("Request to Read counter: %s...", name) + + try: + counter = Counter.find(name) + except DatabaseConnectionError as err: + abort(status.HTTP_503_SERVICE_UNAVAILABLE, err) - Args: - name (str): the name of the counter to create + if not counter: + abort(status.HTTP_404_NOT_FOUND, f"Counter {name} does not exist") + + app.logger.info("Returning: %d...", counter.value) + return jsonify(counter.serialize()) - Returns: - dict: the counter and it's value - """ - app.logger.info("Request to Create counter %s...", name) - # See if the counter already exists and send an error if it does - counter = Counter.find(name) - if counter is not None: - abort(status.HTTP_409_CONFLICT, f"Counter {name} already exists") +############################################################ +# Create counter +############################################################ +@app.route("/counters/", methods=["POST"]) +def create_counters(name): + """Create a counter""" + app.logger.info("Request to Create counter...") + try: + counter = Counter.find(name) + if counter is not None: + return jsonify(code=status.HTTP_409_CONFLICT, error="Counter already exists"), status.HTTP_409_CONFLICT - # Create the new counter - counter = Counter(name) - counter.create() + counter = Counter(name) + except DatabaseConnectionError as err: + abort(status.HTTP_503_SERVICE_UNAVAILABLE, err) - # Set the location header and return the new counter location_url = url_for("read_counters", name=name, _external=True) return ( jsonify(counter.serialize()), @@ -98,102 +106,37 @@ def create_counters(name): ) -# ----------------------------------------------------------- -# Read counters -# ----------------------------------------------------------- -@app.route("/counters/", methods=["GET"]) -def read_counters(name): - """Reads a counter from the database - - Args: - name (str): the name of the counter to read - - Returns: - dict: the counter and it's value - """ - app.logger.info("Request to Read counter %s...", name) - - # Get the current counter - counter = Counter.find(name) - if counter is None: - abort(status.HTTP_404_NOT_FOUND, f"Counter {name} does not exist") - - # Return the counter - return jsonify(counter.serialize()) - - -# ----------------------------------------------------------- +############################################################ # Update counters -# ----------------------------------------------------------- +############################################################ @app.route("/counters/", methods=["PUT"]) def update_counters(name): - """Updates a counter in the database + """Update a counter""" + app.logger.info("Request to Update counter...") + try: + counter = Counter.find(name) + if counter is None: + return jsonify(code=status.HTTP_404_NOT_FOUND, error=f"Counter {name} does not exist"), status.HTTP_404_NOT_FOUND - Args: - name (str): the name of the counter to update + count = counter.increment() + except DatabaseConnectionError as err: + abort(status.HTTP_503_SERVICE_UNAVAILABLE, err) - Returns: - dict: the counter and it's value - """ - app.logger.info("Request to Update counter %s...", name) - - # Get the current counter - counter = Counter.find(name) - if counter is None: - abort(status.HTTP_404_NOT_FOUND, f"Counter {name} does not exist") - - # Increment the counter and return the new value - counter.update() - - return jsonify(counter.serialize()) + return jsonify(name=name, counter=count) -# ----------------------------------------------------------- +############################################################ # Delete counters -# ----------------------------------------------------------- +############################################################ @app.route("/counters/", methods=["DELETE"]) def delete_counters(name): - """Delete a counter from the database - - Args: - name (str): the name of the counter to delete - - Returns: - str: always returns an empty string - """ - app.logger.info("Request to Delete counter %s...", name) - - # Get the current counter - counter = Counter.find(name) - # If it exists delete it, if not do nothing - if counter is not None: - counter.delete() + """Delete a counter""" + app.logger.info("Request to Delete counter...") + try: + counter = Counter.find(name) + if counter: + del counter.value + except DatabaseConnectionError as err: + abort(status.HTTP_503_SERVICE_UNAVAILABLE, err) - # Delete always returns 204 return "", status.HTTP_204_NO_CONTENT - - -# ----------------------------------------------------------- -# Reset counters action -# ----------------------------------------------------------- -@app.route("/counters//reset", methods=["PUT"]) -def reset_counters(name): - """Resets a counter back to zero - - Args: - name (str): the name of the counter to reset - - Returns: - dict: the counter and it's zero value - """ - app.logger.info("Request to Reset counter %s...", name) - - # Get the current counter - counter = Counter.find(name) - if counter is None: - abort(status.HTTP_404_NOT_FOUND, f"Counter {name} does not exist") - - # reset the counter to zero - counter.reset() - - return jsonify(counter.serialize()) diff --git a/tests/test_models.py b/tests/test_models.py index 92546c1..622bc7f 100644 --- a/tests/test_models.py +++ b/tests/test_models.py @@ -1,116 +1,146 @@ +# -*- coding: utf-8 -*- +# Copyright 2016, 2021 John J. Rofrano. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + +# pylint: disable=disallowed-name """ -Test cases for YourResourceModel Model +Test cases for Counter Model +Test cases can be run with the following: + nosetests -v --with-spec --spec-color + coverage report -m """ import os import logging -import unittest -from service import app -from service.models import Counter +from unittest import TestCase +from unittest.mock import patch +from redis.exceptions import ConnectionError as RedisConnectionError +from service.models import Counter, DatabaseConnectionError + +DATABASE_URI = os.getenv("DATABASE_URI", "redis://:@localhost:6379/0") + +logging.disable(logging.CRITICAL) -# Get the database from the environment (12 factor) -DATABASE_URI = os.getenv("DATABASE_URI", "redis://localhost:6379") -TEST_COUNTER = "foo" ###################################################################### -# C O U N T E R M O D E L T E S T C A S E S +# T E S T C A S E S ###################################################################### -class TestCounterModel(unittest.TestCase): - """ Test Cases for Counter Model """ +class CounterTests(TestCase): + """Counter Model Tests""" @classmethod def setUpClass(cls): - """ This runs once before the entire test suite """ - app.config["TESTING"] = True - app.config["DEBUG"] = False - app.config["DATABASE_URI"] = DATABASE_URI - app.logger.setLevel(logging.CRITICAL) - Counter.init_db(app) - - @classmethod - def tearDownClass(cls): - """ This runs once after the entire test suite """ - pass + """Run before all tests""" + # Counter.connect(DATABASE_URI) def setUp(self): - """ This runs before each test """ + """This runs before each test""" + Counter.connect(DATABASE_URI) Counter.remove_all() + self.counter = Counter() def tearDown(self): - """ This runs after each test """ - pass + """This runs after each test""" + # Counter.redis.flushall() ###################################################################### # T E S T C A S E S ###################################################################### - def test_create_a_counter(self): - """ It should create a counter """ - counter = Counter(TEST_COUNTER) - counter.create() - self.assertEqual(counter.name, TEST_COUNTER) - self.assertEqual(counter.count, 0) - - def test_read_a_counter(self): - """ It should read a counter """ - counter = Counter(TEST_COUNTER) - counter.create() - results = Counter.find(TEST_COUNTER) - self.assertEqual(results.name, TEST_COUNTER) - self.assertEqual(results.count, 0) - - def test_update_a_counter(self): - """ It should update a counter """ - counter = Counter(TEST_COUNTER) - counter.create() - counter.update() - self.assertEqual(counter.name, TEST_COUNTER) - self.assertEqual(counter.count, 1) - - def test_list_a_counter(self): - """ It should list counters """ - Counter(TEST_COUNTER).create() - Counter("bar").create() - Counter("baz").create() - results = Counter.all() - self.assertEqual(len(results), 3) - - def test_delete_a_counter(self): - """ It should delete a counter """ - counter = Counter(TEST_COUNTER) - counter.create() - results = Counter.find(TEST_COUNTER) - self.assertEqual(results.name, TEST_COUNTER) - counter.delete() - results = Counter.find(TEST_COUNTER) - self.assertIsNone(results) - - def test_reset_a_counter(self): - """ It should reset a counter """ - counter = Counter(TEST_COUNTER) - counter.create() - # update it to 2 - counter.update() - counter.update() - self.assertEqual(counter.name, TEST_COUNTER) - self.assertEqual(counter.count, 2) - # reset the counter - counter.reset() - # assert that the in-memory object was reset - self.assertEqual(counter.count, 0) - # fetch it from the database and check again - results = Counter.find(TEST_COUNTER) - self.assertEqual(results.name, TEST_COUNTER) - self.assertEqual(counter.count, 0) - - def test_serialize_a_counter(self): - """ It should serialize a counter """ - counter = Counter(TEST_COUNTER) - results = counter.serialize() - self.assertEqual(results["name"], TEST_COUNTER) - self.assertEqual(results["count"], 0) - - def test_repr_a_counter(self): - """ It should represent a counter """ - counter = Counter(TEST_COUNTER) - self.assertEqual(str(counter), f"") + def test_create_counter_with_name(self): + """It should Create a counter with a name""" + counter = Counter("foo") + self.assertIsNotNone(counter) + self.assertEqual(counter.name, "foo") + self.assertEqual(counter.value, 0) + + def test_create_counter_no_name(self): + """It should not Create a counter without a name""" + self.assertIsNotNone(self.counter) + self.assertEqual(self.counter.name, "hits") + self.assertEqual(self.counter.value, 0) + + def test_serialize_counter(self): + """It should Serialize a counter""" + self.assertIsNotNone(self.counter) + data = self.counter.serialize() + self.assertEqual(data["name"], "hits") + self.assertEqual(data["counter"], 0) + + def test_set_list_counters(self): + """It should List all of the counters""" + _ = Counter("foo") + _ = Counter("bar") + counters = Counter.all() + self.assertEqual(len(counters), 3) + + def test_set_find_counter(self): + """It should Find a counter""" + _ = Counter("foo") + _ = Counter("bar") + foo = Counter.find("foo") + self.assertEqual(foo.name, "foo") + + def test_counter_not_found(self): + """It should not find a counter""" + foo = Counter.find("foo") + self.assertIsNone(foo) + + def test_set_get_counter(self): + """It should Set and then Get the counter""" + self.counter.value = 13 + self.assertEqual(self.counter.value, 13) + + def test_delete_counter(self): + """It should Delete a counter""" + counter = Counter("foo") + self.assertEqual(counter.value, 0) + del counter.value + found = Counter.find("foo") + self.assertIsNone(found) + + self.assertEqual(self.counter.value, 0) + + def test_increment_counter(self): + """It should Increment the current value of the counter by 1""" + count = self.counter.value + next_count = self.counter.increment() + logging.debug( + "count(%s) = %s, next_count(%s) = %s", + type(count), + count, + type(next_count), + next_count, + ) + self.assertEqual(next_count, count + 1) + + def test_increment_counter_to_2(self): + """It should Increment the counter to 2""" + self.assertEqual(self.counter.value, 0) + self.counter.increment() + self.assertEqual(self.counter.value, 1) + counter = Counter.find("hits") + counter.increment() + self.assertEqual(counter.value, 2) + + @patch("redis.Redis.ping") + def test_no_connection(self, ping_mock): + """It should Handle a failed connection""" + ping_mock.side_effect = RedisConnectionError() + self.assertRaises(DatabaseConnectionError, self.counter.connect, DATABASE_URI) + + @patch.dict(os.environ, {"DATABASE_URI": ""}) + def test_missing_environment_creds(self): + """It should detect Missing environment credentials""" + self.assertRaises(DatabaseConnectionError, self.counter.connect) diff --git a/tests/test_routes.py b/tests/test_routes.py index 13c1e92..5a8ca1c 100644 --- a/tests/test_routes.py +++ b/tests/test_routes.py @@ -1,5 +1,20 @@ +# -*- coding: utf-8 -*- +# Copyright status.HTTP_201_CREATED6, 2020 John J. Rofrano. All Rights Reserved. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. + """ -TestYourResourceModel API Service Test Suite +Counter API Service Test Suite Test cases can be run with the following: nosetests -v --with-spec --spec-color @@ -8,133 +23,160 @@ import os import logging from unittest import TestCase -from unittest.mock import MagicMock, patch -from service import app -from service.models import Counter -from service.common import status # HTTP Status Codes +from unittest.mock import patch +from wsgi import app +from service.models import Counter, DatabaseConnectionError +from service.common import status + +# logging.disable(logging.CRITICAL) + +DATABASE_URI = os.getenv("DATABASE_URI", "redis://:@localhost:6379/0") -TEST_COUNTER = "foo" ###################################################################### # T E S T C A S E S ###################################################################### -class TestYourResourceServer(TestCase): - """ REST API Server Tests """ +class ServiceTest(TestCase): + """REST API Server Tests""" @classmethod def setUpClass(cls): - """ This runs once before the entire test suite """ + """This runs once before the entire test suite""" app.testing = True - app.logger.setLevel(logging.CRITICAL) + app.debug = False @classmethod def tearDownClass(cls): - """ This runs once after the entire test suite """ - pass + """This runs once after the entire test suite""" def setUp(self): - """ This runs before each test """ + """This runs before each test""" + Counter.connect(DATABASE_URI) Counter.remove_all() self.app = app.test_client() def tearDown(self): - """ This runs after each test """ - pass + """This runs after each test""" -###################################################################### -# T E S T C A S E S -###################################################################### + ###################################################################### + # T E S T C A S E S + ###################################################################### def test_index(self): - """ It should call the home page """ + """It should return the home page""" resp = self.app.get("/") self.assertEqual(resp.status_code, status.HTTP_200_OK) - def test_create_counters(self): - """ It should Create a counter """ - resp = self.app.post(f"/counters/{TEST_COUNTER}") + def test_create_counter(self): + """It should Create a counter""" + resp = self.app.post("/counters/foo") self.assertEqual(resp.status_code, status.HTTP_201_CREATED) data = resp.get_json() - self.assertEqual(data["name"], TEST_COUNTER) - self.assertEqual(data["count"], 0) + self.assertEqual(data["counter"], 0) + + def test_counter_already_exists(self): + """It should not Counter that already exists""" + resp = self.app.post("/counters/foo") + self.assertEqual(resp.status_code, status.HTTP_201_CREATED) + resp = self.app.post("/counters/foo") + self.assertEqual(resp.status_code, status.HTTP_409_CONFLICT) def test_list_counters(self): - """ It should List counters """ - resp = self.app.get("/counters") - self.assertEqual(resp.status_code, status.HTTP_200_OK) - data = resp.get_json() - self.assertEqual(len(data), 0) - # create a counter and name sure it appears in the list - self.app.post("/counters/foo") + """It should Get multiple counters""" + resp = self.app.post("/counters/foo") + self.assertEqual(resp.status_code, status.HTTP_201_CREATED) + resp = self.app.post("/counters/bar") + self.assertEqual(resp.status_code, status.HTTP_201_CREATED) resp = self.app.get("/counters") self.assertEqual(resp.status_code, status.HTTP_200_OK) data = resp.get_json() - self.assertEqual(len(data), 1) + self.assertEqual(len(data), 2) - def test_read_counters(self): - """ It should Read a counter """ - self.test_create_counters() - resp = self.app.get(f"/counters/{TEST_COUNTER}") + def test_get_counter(self): + """It should Get a counter""" + self.test_create_counter() + resp = self.app.get("/counters/foo") self.assertEqual(resp.status_code, status.HTTP_200_OK) data = resp.get_json() - self.assertEqual(data["name"], TEST_COUNTER) - self.assertEqual(data["count"], 0) - - def test_update_counters(self): - """ It should Update a counter """ - self.test_read_counters() - # now update it - resp = self.app.put(f"/counters/{TEST_COUNTER}") - self.assertEqual(resp.status_code, status.HTTP_200_OK) - data = resp.get_json() - self.assertEqual(data["name"], TEST_COUNTER) - self.assertEqual(data["count"], 1) + self.assertEqual(data["counter"], 0) - def test_delete_counters(self): - """ It should Delete a counter """ - self.test_create_counters() - resp = self.app.delete(f"/counters/{TEST_COUNTER}") - self.assertEqual(resp.status_code, status.HTTP_204_NO_CONTENT) - resp = self.app.get(f"/counters/{TEST_COUNTER}") + def test_get_counter_not_found(self): + """It should not return a counter that does not exist""" + resp = self.app.get("/counters/foo") self.assertEqual(resp.status_code, status.HTTP_404_NOT_FOUND) - def test_counter_already_exists(self): - """ It should detect counter already exists """ - self.test_create_counters() - resp = self.app.post(f"/counters/{TEST_COUNTER}") - self.assertEqual(resp.status_code, status.HTTP_409_CONFLICT) - - def test_update_unknown_counter(self): - """ It should not Update a counter that doesn't exist """ - resp = self.app.put("/counters/bar") + def test_put_counter_not_found(self): + """It should not update a counter that does not exist""" + resp = self.app.put("/counters/foo") self.assertEqual(resp.status_code, status.HTTP_404_NOT_FOUND) - def test_reset_counter(self): - """ It should Reset a counter """ - self.test_create_counters() - # update counter to 3 - resp = self.app.put(f"/counters/{TEST_COUNTER}") - self.assertEqual(resp.status_code, status.HTTP_200_OK) - resp = self.app.put(f"/counters/{TEST_COUNTER}") - self.assertEqual(resp.status_code, status.HTTP_200_OK) - resp = self.app.put(f"/counters/{TEST_COUNTER}") + def test_increment_counter(self): + """It should Increment the counter""" + self.test_get_counter() + resp = self.app.put("/counters/foo") self.assertEqual(resp.status_code, status.HTTP_200_OK) data = resp.get_json() - self.assertEqual(data["name"], TEST_COUNTER) - self.assertEqual(data["count"], 3) - # reset counter to zero - resp = self.app.put(f"/counters/{TEST_COUNTER}/reset") + self.assertEqual(data["counter"], 1) + + resp = self.app.put("/counters/foo") self.assertEqual(resp.status_code, status.HTTP_200_OK) data = resp.get_json() - self.assertEqual(data["name"], TEST_COUNTER) - self.assertEqual(data["count"], 0) + logging.debug(data) + self.assertEqual(data["counter"], 2) - def test_reset_unknown_counter(self): - """ It should not Reset a counter that doesn't exist """ - resp = self.app.put(f"/counters/{TEST_COUNTER}/reset") - self.assertEqual(resp.status_code, status.HTTP_404_NOT_FOUND) + def test_delete_counter(self): + """It should Delete the counter""" + self.test_create_counter() + resp = self.app.delete("/counters/foo") + self.assertEqual(resp.status_code, status.HTTP_204_NO_CONTENT) - def test_method_not_allowed_handler(self): - """ It should trigger Method Not Allowed error handler """ - resp = self.app.get(f"/counters/{TEST_COUNTER}/reset") + def test_method_not_allowed(self): + """It should not allow usuported Methods""" + resp = self.app.post("/counters") self.assertEqual(resp.status_code, status.HTTP_405_METHOD_NOT_ALLOWED) + + ###################################################################### + # T E S T E R R O R H A N D L E R S + ###################################################################### + + @patch("service.routes.Counter.redis.get") + def test_failed_get_request(self, redis_mock): + """It should handle Error for failed GET""" + redis_mock.return_value = 0 + redis_mock.side_effect = DatabaseConnectionError() + resp = self.app.get("/counters/foo") + self.assertEqual(resp.status_code, status.HTTP_503_SERVICE_UNAVAILABLE) + + @patch("service.models.Counter.increment") + def test_failed_update_request(self, value_mock): + """It should handle Error for failed UPDATE""" + value_mock.return_value = 0 + value_mock.side_effect = DatabaseConnectionError() + self.test_create_counter() + resp = self.app.put("/counters/foo") + self.assertEqual(resp.status_code, status.HTTP_503_SERVICE_UNAVAILABLE) + + @patch("service.models.Counter.__init__") + def test_failed_post_request(self, value_mock): + """It should handle Error for failed POST""" + value_mock.return_value = 0 + value_mock.side_effect = DatabaseConnectionError() + resp = self.app.post("/counters/foo") + self.assertEqual(resp.status_code, status.HTTP_503_SERVICE_UNAVAILABLE) + + @patch("service.routes.Counter.redis.keys") + def test_failed_list_request(self, redis_mock): + """It should handle Error for failed LIST""" + redis_mock.return_value = 0 + redis_mock.side_effect = Exception() + resp = self.app.get("/counters") + self.assertEqual(resp.status_code, status.HTTP_503_SERVICE_UNAVAILABLE) + + def test_failed_delete_request(self): + """It should handle Error for failed DELETE""" + self.test_create_counter() + with patch("service.routes.Counter.redis.get") as redis_mock: + redis_mock.return_value = 0 + redis_mock.side_effect = DatabaseConnectionError() + resp = self.app.delete("/counters/foo") + self.assertEqual(resp.status_code, status.HTTP_503_SERVICE_UNAVAILABLE) diff --git a/wsgi.py b/wsgi.py new file mode 100644 index 0000000..8de5c57 --- /dev/null +++ b/wsgi.py @@ -0,0 +1,9 @@ +""" +Web Server Gateway Interface (WSGI) entry point +""" +from service import init_app + +app = init_app() + +if __name__ == "__main__": + app.run(host='0.0.0.0') From 09f74522d6e40f0757e44c30abf155a52625b6e3 Mon Sep 17 00:00:00 2001 From: John Rofrano Date: Sun, 4 Feb 2024 17:15:43 +0000 Subject: [PATCH 4/5] Removed unneeded try/catch blocks --- service/common/error_handlers.py | 9 +++- service/routes.py | 81 +++++++++++++++++--------------- 2 files changed, 50 insertions(+), 40 deletions(-) diff --git a/service/common/error_handlers.py b/service/common/error_handlers.py index ce7ed6b..5d31a99 100644 --- a/service/common/error_handlers.py +++ b/service/common/error_handlers.py @@ -1,5 +1,5 @@ ###################################################################### -# Copyright 2016, 2022 John J. Rofrano. All Rights Reserved. +# Copyright 2016, 2024 John J. Rofrano. All Rights Reserved. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. @@ -22,12 +22,19 @@ from flask import jsonify from flask import current_app as app from service.common import status +from service.models import DatabaseConnectionError ###################################################################### # Error Handlers ###################################################################### +@app.errorhandler(DatabaseConnectionError) +def request_validation_error(error): + """Handles Value Errors from bad data""" + return service_unavailable(error) + + @app.errorhandler(status.HTTP_404_NOT_FOUND) def not_found(error): """Handles resources not found with 404_NOT_FOUND""" diff --git a/service/routes.py b/service/routes.py index e727745..47c1897 100644 --- a/service/routes.py +++ b/service/routes.py @@ -19,15 +19,10 @@ This service keeps track of named counters """ - -import os from flask import jsonify, abort, url_for from flask import current_app as app from service.common import status # HTTP Status Codes -from .models import Counter, DatabaseConnectionError - -DEBUG = os.getenv("DEBUG", "False") == "True" -PORT = os.getenv("PORT", "8080") +from service.models import Counter ###################################################################### @@ -46,6 +41,10 @@ def index(): status.HTTP_200_OK, ) +############################################################ +# R E S T A P I M E T H O D S +############################################################ + ############################################################ # List counters @@ -54,11 +53,10 @@ def index(): def list_counters(): """List counters""" app.logger.info("Request to list all counters...") - try: - counters = Counter.all() - except DatabaseConnectionError as err: - abort(status.HTTP_503_SERVICE_UNAVAILABLE, err) + counters = Counter.all() + + app.logger.info("Returning %d counters...", len(counters)) return jsonify(counters) @@ -68,15 +66,12 @@ def list_counters(): @app.route("/counters/", methods=["GET"]) def read_counters(name): """Read a counter""" - app.logger.info("Request to Read counter: %s...", name) + app.logger.info("Request to Read counter: '%s'...", name) - try: - counter = Counter.find(name) - except DatabaseConnectionError as err: - abort(status.HTTP_503_SERVICE_UNAVAILABLE, err) + counter = Counter.find(name) if not counter: - abort(status.HTTP_404_NOT_FOUND, f"Counter {name} does not exist") + error(status.HTTP_404_NOT_FOUND, f"Counter '{name}' does not exist") app.logger.info("Returning: %d...", counter.value) return jsonify(counter.serialize()) @@ -88,17 +83,16 @@ def read_counters(name): @app.route("/counters/", methods=["POST"]) def create_counters(name): """Create a counter""" - app.logger.info("Request to Create counter...") - try: - counter = Counter.find(name) - if counter is not None: - return jsonify(code=status.HTTP_409_CONFLICT, error="Counter already exists"), status.HTTP_409_CONFLICT + app.logger.info("Request to Create counter: '%s'...", name) - counter = Counter(name) - except DatabaseConnectionError as err: - abort(status.HTTP_503_SERVICE_UNAVAILABLE, err) + counter = Counter.find(name) + if counter is not None: + error(status.HTTP_409_CONFLICT, f"Counter '{name}' already exists") + + counter = Counter(name) location_url = url_for("read_counters", name=name, _external=True) + app.logger.info("Counter '%s' created", name) return ( jsonify(counter.serialize()), status.HTTP_201_CREATED, @@ -112,16 +106,15 @@ def create_counters(name): @app.route("/counters/", methods=["PUT"]) def update_counters(name): """Update a counter""" - app.logger.info("Request to Update counter...") - try: - counter = Counter.find(name) - if counter is None: - return jsonify(code=status.HTTP_404_NOT_FOUND, error=f"Counter {name} does not exist"), status.HTTP_404_NOT_FOUND + app.logger.info("Request to Update counter: '%s'...", name) + + counter = Counter.find(name) + if counter is None: + error(status.HTTP_404_NOT_FOUND, f"Counter '{name}' does not exist") - count = counter.increment() - except DatabaseConnectionError as err: - abort(status.HTTP_503_SERVICE_UNAVAILABLE, err) + count = counter.increment() + app.logger.info("Counter '%s' updated to %d", name, count) return jsonify(name=name, counter=count) @@ -131,12 +124,22 @@ def update_counters(name): @app.route("/counters/", methods=["DELETE"]) def delete_counters(name): """Delete a counter""" - app.logger.info("Request to Delete counter...") - try: - counter = Counter.find(name) - if counter: - del counter.value - except DatabaseConnectionError as err: - abort(status.HTTP_503_SERVICE_UNAVAILABLE, err) + app.logger.info("Request to Delete counter: '%s'...", name) + + counter = Counter.find(name) + if counter: + del counter.value + app.logger.info("Counter '%s' deleted", name) return "", status.HTTP_204_NO_CONTENT + + +############################################################ +# U T I L I T Y F U N C T I O N S +############################################################ + + +def error(status_code, reason): + """Logs the error and then aborts""" + app.logger.error(reason) + abort(status_code, reason) From 66b0cacd43c1f23af9c3953d95f3ca1db58d6056 Mon Sep 17 00:00:00 2001 From: John Rofrano Date: Sun, 4 Feb 2024 17:31:42 +0000 Subject: [PATCH 5/5] Updated CI to new dev tools (poetry) --- .github/workflows/ci.yaml | 35 ++++++++++++++++++++--------------- poetry.lock | 2 +- pyproject.toml | 2 +- 3 files changed, 22 insertions(+), 17 deletions(-) diff --git a/.github/workflows/ci.yaml b/.github/workflows/ci.yaml index 753cc69..be6a7ca 100644 --- a/.github/workflows/ci.yaml +++ b/.github/workflows/ci.yaml @@ -3,14 +3,18 @@ on: push: branches: - master + paths-ignore: + - 'README.md' pull_request: branches: - master + paths-ignore: + - 'README.md' jobs: build: runs-on: ubuntu-latest - container: python:3.9-slim + container: python:3.11-slim # Required services services: @@ -18,31 +22,32 @@ jobs: image: redis:6-alpine ports: - 6379:6379 + options: >- + --health-cmd "redis-cli ping" + --health-interval 10s + --health-timeout 5s + --health-retries 5 steps: - name: Checkout code - uses: actions/checkout@v2 - + uses: actions/checkout@v3 + - name: Install Python dependencies run: | - python -m pip install --upgrade pip wheel - pip install -r requirements.txt + python -m pip install --upgrade pip poetry + poetry config virtualenvs.create false + poetry install - name: Linting run: | - # stop the build if there are Python syntax errors or undefined names flake8 service --count --select=E9,F63,F7,F82 --show-source --statistics - # exit-zero treats all errors as warnings. The GitHub editor is 127 chars wide flake8 service --count --max-complexity=10 --max-line-length=127 --statistics + pylint service --max-line-length=127 - - name: Run unit tests with nose - run: nosetests -v --with-spec --spec-color --with-coverage --cover-package=app + - name: Run unit tests with PyTest + run: pytest --pspec --cov=service --cov-fail-under=95 env: - DATABASE_URI: "redis://redis:6379/0" + DATABASE_URI: "redis://redis:6379" - name: Upload code coverage - uses: codecov/codecov-action@v2 - with: - files: ./coverage.xml - flags: unittests - version: "v0.1.13" + uses: codecov/codecov-action@v3.1.4 diff --git a/poetry.lock b/poetry.lock index 52a0284..8f02738 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1111,4 +1111,4 @@ watchdog = ["watchdog (>=2.3)"] [metadata] lock-version = "2.0" python-versions = "^3.11" -content-hash = "cf73b89a8440ba253119d096b18749a41f7f339ce2a7c0d19c9336ba73ca365d" +content-hash = "64d5d2851c5b756a30274ea35d894dd1eaa982ca0f90046624a9c3a60a6f246a" diff --git a/pyproject.toml b/pyproject.toml index 7de6c40..e18ad65 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -9,7 +9,7 @@ readme = "README.md" [tool.poetry.dependencies] python = "^3.11" Flask = "^3.0.2" -redis = "^4.5.3" +redis = "^4.5.4" flask-redis = "^0.4.0" python-dotenv = "^1.0.0" gunicorn = "^21.2.0"