From 6754f04fcd70a05b8f48fb62f44d65a92c8959f4 Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Mon, 3 Aug 2020 21:24:06 +0200 Subject: [PATCH 01/13] Add nox as the task runner - set up skeletons for all planned nox sessions - provide a base configuration for all nox sessions - create two utility functions: + _begin() => run commands common to all sessions + _install_packages() => install dependencies in nox sessions respecting the versions pinned in poetry.lock - add the following file: + noxfile.py --- .gitignore | 1 + noxfile.py | 114 +++++++++++++++++++++++++++++++++++++ poetry.lock | 148 ++++++++++++++++++++++++++++++++++++++++++++++++- pyproject.toml | 2 + 4 files changed, 263 insertions(+), 2 deletions(-) create mode 100644 noxfile.py diff --git a/.gitignore b/.gitignore index 47d1330..2e72e0b 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,4 @@ +.cache/ *.egg-info/ .python-version .venv/ diff --git a/noxfile.py b/noxfile.py new file mode 100644 index 0000000..14b844b --- /dev/null +++ b/noxfile.py @@ -0,0 +1,114 @@ +"""Configure nox for the isolated test and lint environments.""" + +import os +import tempfile +from typing import Any + +import nox +from nox.sessions import Session + + +MAIN_PYTHON = '3.8' + +# Keep the project is forward compatible. +NEXT_PYTHON = '3.9' + +# Paths with *.py files. +SRC_LOCATIONS = 'noxfile.py', 'src/' + + +# Use a unified .cache/ folder for all tools. +nox.options.envdir = '.cache/nox' + +# All tools except git and poetry are project dependencies. +# Avoid accidental successes if the environment is not set up properly. +nox.options.error_on_external_run = True + +# Run only CI related checks by default. +nox.options.sessions = ( + 'format', + 'lint', + f'test-{MAIN_PYTHON}', + f'test-{NEXT_PYTHON}', +) + + +@nox.session(name='format', python=MAIN_PYTHON) +def format_(session: Session): + """Format source files.""" + _begin(session) + + +@nox.session(python=MAIN_PYTHON) +def lint(session: Session): + """Lint source files.""" + _begin(session) + + +@nox.session(python=[MAIN_PYTHON, NEXT_PYTHON]) +def test(session: Session): + """Test the code base.""" + _begin(session) + + +@nox.session(name='pre-commit', python=MAIN_PYTHON, venv_backend='none') +def pre_commit(session: Session): + """Source files must be well-formed before they enter git.""" + _begin(session) + session.notify('format') + session.notify('lint') + + +@nox.session(name='pre-merge', python=MAIN_PYTHON, venv_backend='none') +def pre_merge(session: Session): + """The test suite must pass before merges are made.""" + _begin(session) + session.notify('test') + + +def _begin(session: Session) -> None: + """Show generic info about a session.""" + if session.posargs: + print('extra arguments:', *session.posargs) + + session.run('python', '--version') + + # Fake GNU's pwd. + session.log('pwd') + print(os.getcwd()) + + +def _install_packages( + session: Session, *packages_or_pip_args: str, **kwargs: Any +) -> None: + """Install packages respecting the poetry.lock file. + + This function wraps nox.sessions.Session.install() such that it installs + packages respecting the pinnned versions specified in poetry's lock file. + This makes nox sessions even more deterministic. + + Args: + session: the Session object + *packages_or_pip_args: the packages to be installed or pip options + **kwargs: passed on to nox.sessions.Session.install() + """ + if session.virtualenv.reuse_existing: + session.log( + 'No dependencies are installed as an existing environment is re-used', + ) + return + + session.log('Dependencies are installed respecting the poetry.lock file') + + with tempfile.NamedTemporaryFile() as requirements_txt: + session.run( + 'poetry', + 'export', + '--dev', + '--format=requirements.txt', + f'--output={requirements_txt.name}', + external=True, + ) + session.install( + f'--constraint={requirements_txt.name}', *packages_or_pip_args, **kwargs, + ) diff --git a/poetry.lock b/poetry.lock index b9e1c09..d1f77fe 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,8 +1,152 @@ -package = [] +[[package]] +category = "dev" +description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." +name = "appdirs" +optional = false +python-versions = "*" +version = "1.4.4" + +[[package]] +category = "dev" +description = "Bash tab completion for argparse" +name = "argcomplete" +optional = false +python-versions = "*" +version = "1.12.0" + +[package.extras] +test = ["coverage", "flake8", "pexpect", "wheel"] + +[[package]] +category = "dev" +description = "Cross-platform colored terminal text." +marker = "sys_platform == \"win32\"" +name = "colorama" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +version = "0.4.3" + +[[package]] +category = "dev" +description = "Log formatting with colors!" +name = "colorlog" +optional = false +python-versions = "*" +version = "4.2.1" + +[package.dependencies] +colorama = "*" + +[[package]] +category = "dev" +description = "Distribution utilities" +name = "distlib" +optional = false +python-versions = "*" +version = "0.3.1" + +[[package]] +category = "dev" +description = "A platform independent file lock." +name = "filelock" +optional = false +python-versions = "*" +version = "3.0.12" + +[[package]] +category = "dev" +description = "Flexible test automation." +name = "nox" +optional = false +python-versions = ">=3.5" +version = "2020.5.24" + +[package.dependencies] +argcomplete = ">=1.9.4,<2.0" +colorlog = ">=2.6.1,<5.0.0" +py = ">=1.4.0,<2.0.0" +virtualenv = ">=14.0.0" + +[package.extras] +tox_to_nox = ["jinja2", "tox"] + +[[package]] +category = "dev" +description = "library with cross-python path, ini-parsing, io, code, log facilities" +name = "py" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "1.9.0" + +[[package]] +category = "dev" +description = "Python 2 and 3 compatibility utilities" +name = "six" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" +version = "1.15.0" + +[[package]] +category = "dev" +description = "Virtual Python Environment builder" +name = "virtualenv" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,>=2.7" +version = "20.0.29" + +[package.dependencies] +appdirs = ">=1.4.3,<2" +distlib = ">=0.3.1,<1" +filelock = ">=3.0.0,<4" +six = ">=1.9.0,<2" + +[package.extras] +docs = ["proselint (>=0.10.2)", "sphinx (>=3)", "sphinx-argparse (>=0.2.5)", "sphinx-rtd-theme (>=0.4.3)", "towncrier (>=19.9.0rc1)"] +testing = ["coverage (>=5)", "coverage-enable-subprocess (>=1)", "flaky (>=3)", "pytest (>=4)", "pytest-env (>=0.6.2)", "pytest-freezegun (>=0.4.1)", "pytest-mock (>=2)", "pytest-randomly (>=1)", "pytest-timeout (>=1)", "pytest-xdist (>=1.31.0)", "packaging (>=20.0)", "xonsh (>=0.9.16)"] [metadata] -content-hash = "fafb334cb038533f851c23d0b63254223abf72ce4f02987e7064b0c95566699a" +content-hash = "4ff12f695975572453949608224e06fb36641151ee69668f8603f7bc3435a51f" lock-version = "1.0" python-versions = "^3.8" [metadata.files] +appdirs = [ + {file = "appdirs-1.4.4-py2.py3-none-any.whl", hash = "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128"}, + {file = "appdirs-1.4.4.tar.gz", hash = "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41"}, +] +argcomplete = [ + {file = "argcomplete-1.12.0-py2.py3-none-any.whl", hash = "sha256:91dc7f9c7f6281d5a0dce5e73d2e33283aaef083495c13974a7dd197a1cdc949"}, + {file = "argcomplete-1.12.0.tar.gz", hash = "sha256:2fbe5ed09fd2c1d727d4199feca96569a5b50d44c71b16da9c742201f7cc295c"}, +] +colorama = [ + {file = "colorama-0.4.3-py2.py3-none-any.whl", hash = "sha256:7d73d2a99753107a36ac6b455ee49046802e59d9d076ef8e47b61499fa29afff"}, + {file = "colorama-0.4.3.tar.gz", hash = "sha256:e96da0d330793e2cb9485e9ddfd918d456036c7149416295932478192f4436a1"}, +] +colorlog = [ + {file = "colorlog-4.2.1-py2.py3-none-any.whl", hash = "sha256:43597fd822ce705190fc997519342fdaaf44b9b47f896ece7aa153ed4b909c74"}, + {file = "colorlog-4.2.1.tar.gz", hash = "sha256:75e55822c3a3387d721579241e776de2cf089c9ef9528b1f09e8b04d403ad118"}, +] +distlib = [ + {file = "distlib-0.3.1-py2.py3-none-any.whl", hash = "sha256:8c09de2c67b3e7deef7184574fc060ab8a793e7adbb183d942c389c8b13c52fb"}, + {file = "distlib-0.3.1.zip", hash = "sha256:edf6116872c863e1aa9d5bb7cb5e05a022c519a4594dc703843343a9ddd9bff1"}, +] +filelock = [ + {file = "filelock-3.0.12-py3-none-any.whl", hash = "sha256:929b7d63ec5b7d6b71b0fa5ac14e030b3f70b75747cef1b10da9b879fef15836"}, + {file = "filelock-3.0.12.tar.gz", hash = "sha256:18d82244ee114f543149c66a6e0c14e9c4f8a1044b5cdaadd0f82159d6a6ff59"}, +] +nox = [ + {file = "nox-2020.5.24-py3-none-any.whl", hash = "sha256:c4509621fead99473a1401870e680b0aadadce5c88440f0532863595176d64c1"}, + {file = "nox-2020.5.24.tar.gz", hash = "sha256:61a55705736a1a73efbd18d5b262a43d55a1176546e0eb28b29064cfcffe26c0"}, +] +py = [ + {file = "py-1.9.0-py2.py3-none-any.whl", hash = "sha256:366389d1db726cd2fcfc79732e75410e5fe4d31db13692115529d34069a043c2"}, + {file = "py-1.9.0.tar.gz", hash = "sha256:9ca6883ce56b4e8da7e79ac18787889fa5206c79dcc67fb065376cd2fe03f342"}, +] +six = [ + {file = "six-1.15.0-py2.py3-none-any.whl", hash = "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"}, + {file = "six-1.15.0.tar.gz", hash = "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259"}, +] +virtualenv = [ + {file = "virtualenv-20.0.29-py2.py3-none-any.whl", hash = "sha256:8aa9c37b082664dbce2236fa420759c02d64109d8e6013593ad13914718a30fd"}, + {file = "virtualenv-20.0.29.tar.gz", hash = "sha256:f14a0a98ea4397f0d926cff950361766b6a73cd5975ae7eb259d12919f819a25"}, +] diff --git a/pyproject.toml b/pyproject.toml index eb24353..158fb5d 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -23,3 +23,5 @@ repository = "https://github.com/webartifex/urban-meal-delivery" python = "^3.8" [tool.poetry.dev-dependencies] +# Task Runners +nox = "^2020.5.24" From bb6de0570928dc7fbcef434a3c488325c0c16b88 Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Mon, 3 Aug 2020 21:39:49 +0200 Subject: [PATCH 02/13] Add a code formatting tool chain - (auto-)format code with: + autoflake => * remove unused imports and variables * remove duplicate dict keys * expand star imports + black => enforce an uncompromising code style + isort => enforce a consistent import style (complying with Google's Python Style Guide) - implement the nox session "format" that runs all these tools - add the following file: + setup.cfg => holds configurations for the develop tools --- noxfile.py | 21 +++++- poetry.lock | 185 ++++++++++++++++++++++++++++++++++++++++++++++++- pyproject.toml | 10 +++ setup.cfg | 26 +++++++ 4 files changed, 240 insertions(+), 2 deletions(-) create mode 100644 setup.cfg diff --git a/noxfile.py b/noxfile.py index 14b844b..2a72dfe 100644 --- a/noxfile.py +++ b/noxfile.py @@ -35,8 +35,27 @@ nox.options.sessions = ( @nox.session(name='format', python=MAIN_PYTHON) def format_(session: Session): - """Format source files.""" + """Format source files with autoflake, black, and isort.""" _begin(session) + _install_packages(session, 'autoflake', 'black', 'isort') + # Interpret extra arguments as locations of source files. + locations = session.posargs or SRC_LOCATIONS + session.run('autoflake', '--version') + session.run( + 'autoflake', + '--in-place', + '--recursive', + '--expand-star-imports', + '--remove-all-unused-imports', + '--ignore-init-module-imports', # modifies --remove-all-unused-imports + '--remove-duplicate-keys', + '--remove-unused-variables', + *locations, + ) + session.run('black', '--version') + session.run('black', *locations) + session.run('isort', '--version') + session.run('isort', *locations) @nox.session(python=MAIN_PYTHON) diff --git a/poetry.lock b/poetry.lock index d1f77fe..ffceece 100644 --- a/poetry.lock +++ b/poetry.lock @@ -17,6 +17,59 @@ version = "1.12.0" [package.extras] test = ["coverage", "flake8", "pexpect", "wheel"] +[[package]] +category = "dev" +description = "Classes Without Boilerplate" +name = "attrs" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "19.3.0" + +[package.extras] +azure-pipelines = ["coverage", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface", "pytest-azurepipelines"] +dev = ["coverage", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface", "sphinx", "pre-commit"] +docs = ["sphinx", "zope.interface"] +tests = ["coverage", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface"] + +[[package]] +category = "dev" +description = "Removes unused imports and unused variables" +name = "autoflake" +optional = false +python-versions = "*" +version = "1.3.1" + +[package.dependencies] +pyflakes = ">=1.1.0" + +[[package]] +category = "dev" +description = "The uncompromising code formatter." +name = "black" +optional = false +python-versions = ">=3.6" +version = "19.10b0" + +[package.dependencies] +appdirs = "*" +attrs = ">=18.1.0" +click = ">=6.5" +pathspec = ">=0.6,<1" +regex = "*" +toml = ">=0.9.4" +typed-ast = ">=1.4.0" + +[package.extras] +d = ["aiohttp (>=3.3.2)", "aiohttp-cors"] + +[[package]] +category = "dev" +description = "Composable command line interface toolkit" +name = "click" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +version = "7.1.2" + [[package]] category = "dev" description = "Cross-platform colored terminal text." @@ -53,6 +106,19 @@ optional = false python-versions = "*" version = "3.0.12" +[[package]] +category = "dev" +description = "A Python utility / library to sort Python imports." +name = "isort" +optional = false +python-versions = ">=3.6,<4.0" +version = "5.2.2" + +[package.extras] +colors = ["colorama (>=0.4.3,<0.5.0)"] +pipfile_deprecated_finder = ["pipreqs", "requirementslib", "tomlkit (>=0.5.3)"] +requirements_deprecated_finder = ["pipreqs", "pip-api"] + [[package]] category = "dev" description = "Flexible test automation." @@ -70,6 +136,14 @@ virtualenv = ">=14.0.0" [package.extras] tox_to_nox = ["jinja2", "tox"] +[[package]] +category = "dev" +description = "Utility library for gitignore style pattern matching of file paths." +name = "pathspec" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +version = "0.8.0" + [[package]] category = "dev" description = "library with cross-python path, ini-parsing, io, code, log facilities" @@ -78,6 +152,22 @@ optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" version = "1.9.0" +[[package]] +category = "dev" +description = "passive checker of Python programs" +name = "pyflakes" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "2.2.0" + +[[package]] +category = "dev" +description = "Alternative regular expression module, to replace re." +name = "regex" +optional = false +python-versions = "*" +version = "2020.7.14" + [[package]] category = "dev" description = "Python 2 and 3 compatibility utilities" @@ -86,6 +176,22 @@ optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" version = "1.15.0" +[[package]] +category = "dev" +description = "Python Library for Tom's Obvious, Minimal Language" +name = "toml" +optional = false +python-versions = "*" +version = "0.10.1" + +[[package]] +category = "dev" +description = "a fork of Python 2 and 3 ast modules with type comment support" +name = "typed-ast" +optional = false +python-versions = "*" +version = "1.4.1" + [[package]] category = "dev" description = "Virtual Python Environment builder" @@ -105,7 +211,7 @@ docs = ["proselint (>=0.10.2)", "sphinx (>=3)", "sphinx-argparse (>=0.2.5)", "sp testing = ["coverage (>=5)", "coverage-enable-subprocess (>=1)", "flaky (>=3)", "pytest (>=4)", "pytest-env (>=0.6.2)", "pytest-freezegun (>=0.4.1)", "pytest-mock (>=2)", "pytest-randomly (>=1)", "pytest-timeout (>=1)", "pytest-xdist (>=1.31.0)", "packaging (>=20.0)", "xonsh (>=0.9.16)"] [metadata] -content-hash = "4ff12f695975572453949608224e06fb36641151ee69668f8603f7bc3435a51f" +content-hash = "9894f7290cbab39b1fa0f3d0ba9cd2aebe16228169867415a69e039ee2b12425" lock-version = "1.0" python-versions = "^3.8" @@ -118,6 +224,21 @@ argcomplete = [ {file = "argcomplete-1.12.0-py2.py3-none-any.whl", hash = "sha256:91dc7f9c7f6281d5a0dce5e73d2e33283aaef083495c13974a7dd197a1cdc949"}, {file = "argcomplete-1.12.0.tar.gz", hash = "sha256:2fbe5ed09fd2c1d727d4199feca96569a5b50d44c71b16da9c742201f7cc295c"}, ] +attrs = [ + {file = "attrs-19.3.0-py2.py3-none-any.whl", hash = "sha256:08a96c641c3a74e44eb59afb61a24f2cb9f4d7188748e76ba4bb5edfa3cb7d1c"}, + {file = "attrs-19.3.0.tar.gz", hash = "sha256:f7b7ce16570fe9965acd6d30101a28f62fb4a7f9e926b3bbc9b61f8b04247e72"}, +] +autoflake = [ + {file = "autoflake-1.3.1.tar.gz", hash = "sha256:680cb9dade101ed647488238ccb8b8bfb4369b53d58ba2c8cdf7d5d54e01f95b"}, +] +black = [ + {file = "black-19.10b0-py36-none-any.whl", hash = "sha256:1b30e59be925fafc1ee4565e5e08abef6b03fe455102883820fe5ee2e4734e0b"}, + {file = "black-19.10b0.tar.gz", hash = "sha256:c2edb73a08e9e0e6f65a0e6af18b059b8b1cdd5bef997d7a0b181df93dc81539"}, +] +click = [ + {file = "click-7.1.2-py2.py3-none-any.whl", hash = "sha256:dacca89f4bfadd5de3d7489b7c8a566eee0d3676333fbb50030263894c38c0dc"}, + {file = "click-7.1.2.tar.gz", hash = "sha256:d2b5255c7c6349bc1bd1e59e08cd12acbbd63ce649f2588755783aa94dfb6b1a"}, +] colorama = [ {file = "colorama-0.4.3-py2.py3-none-any.whl", hash = "sha256:7d73d2a99753107a36ac6b455ee49046802e59d9d076ef8e47b61499fa29afff"}, {file = "colorama-0.4.3.tar.gz", hash = "sha256:e96da0d330793e2cb9485e9ddfd918d456036c7149416295932478192f4436a1"}, @@ -134,18 +255,80 @@ filelock = [ {file = "filelock-3.0.12-py3-none-any.whl", hash = "sha256:929b7d63ec5b7d6b71b0fa5ac14e030b3f70b75747cef1b10da9b879fef15836"}, {file = "filelock-3.0.12.tar.gz", hash = "sha256:18d82244ee114f543149c66a6e0c14e9c4f8a1044b5cdaadd0f82159d6a6ff59"}, ] +isort = [ + {file = "isort-5.2.2-py3-none-any.whl", hash = "sha256:aea484023188ef1c38256dd24afa96e914adafe3a911a1786800a74e433006d1"}, + {file = "isort-5.2.2.tar.gz", hash = "sha256:96b27045e3187b9bdde001143b79f9b10a462f372bff7062302818013b6c86f3"}, +] nox = [ {file = "nox-2020.5.24-py3-none-any.whl", hash = "sha256:c4509621fead99473a1401870e680b0aadadce5c88440f0532863595176d64c1"}, {file = "nox-2020.5.24.tar.gz", hash = "sha256:61a55705736a1a73efbd18d5b262a43d55a1176546e0eb28b29064cfcffe26c0"}, ] +pathspec = [ + {file = "pathspec-0.8.0-py2.py3-none-any.whl", hash = "sha256:7d91249d21749788d07a2d0f94147accd8f845507400749ea19c1ec9054a12b0"}, + {file = "pathspec-0.8.0.tar.gz", hash = "sha256:da45173eb3a6f2a5a487efba21f050af2b41948be6ab52b6a1e3ff22bb8b7061"}, +] py = [ {file = "py-1.9.0-py2.py3-none-any.whl", hash = "sha256:366389d1db726cd2fcfc79732e75410e5fe4d31db13692115529d34069a043c2"}, {file = "py-1.9.0.tar.gz", hash = "sha256:9ca6883ce56b4e8da7e79ac18787889fa5206c79dcc67fb065376cd2fe03f342"}, ] +pyflakes = [ + {file = "pyflakes-2.2.0-py2.py3-none-any.whl", hash = "sha256:0d94e0e05a19e57a99444b6ddcf9a6eb2e5c68d3ca1e98e90707af8152c90a92"}, + {file = "pyflakes-2.2.0.tar.gz", hash = "sha256:35b2d75ee967ea93b55750aa9edbbf72813e06a66ba54438df2cfac9e3c27fc8"}, +] +regex = [ + {file = "regex-2020.7.14-cp27-cp27m-win32.whl", hash = "sha256:e46d13f38cfcbb79bfdb2964b0fe12561fe633caf964a77a5f8d4e45fe5d2ef7"}, + {file = "regex-2020.7.14-cp27-cp27m-win_amd64.whl", hash = "sha256:6961548bba529cac7c07af2fd4d527c5b91bb8fe18995fed6044ac22b3d14644"}, + {file = "regex-2020.7.14-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:c50a724d136ec10d920661f1442e4a8b010a4fe5aebd65e0c2241ea41dbe93dc"}, + {file = "regex-2020.7.14-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:8a51f2c6d1f884e98846a0a9021ff6861bdb98457879f412fdc2b42d14494067"}, + {file = "regex-2020.7.14-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:9c568495e35599625f7b999774e29e8d6b01a6fb684d77dee1f56d41b11b40cd"}, + {file = "regex-2020.7.14-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:51178c738d559a2d1071ce0b0f56e57eb315bcf8f7d4cf127674b533e3101f88"}, + {file = "regex-2020.7.14-cp36-cp36m-win32.whl", hash = "sha256:9eddaafb3c48e0900690c1727fba226c4804b8e6127ea409689c3bb492d06de4"}, + {file = "regex-2020.7.14-cp36-cp36m-win_amd64.whl", hash = "sha256:14a53646369157baa0499513f96091eb70382eb50b2c82393d17d7ec81b7b85f"}, + {file = "regex-2020.7.14-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:1269fef3167bb52631ad4fa7dd27bf635d5a0790b8e6222065d42e91bede4162"}, + {file = "regex-2020.7.14-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:d0a5095d52b90ff38592bbdc2644f17c6d495762edf47d876049cfd2968fbccf"}, + {file = "regex-2020.7.14-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:4c037fd14c5f4e308b8370b447b469ca10e69427966527edcab07f52d88388f7"}, + {file = "regex-2020.7.14-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:bc3d98f621898b4a9bc7fecc00513eec8f40b5b83913d74ccb445f037d58cd89"}, + {file = "regex-2020.7.14-cp37-cp37m-win32.whl", hash = "sha256:46bac5ca10fb748d6c55843a931855e2727a7a22584f302dd9bb1506e69f83f6"}, + {file = "regex-2020.7.14-cp37-cp37m-win_amd64.whl", hash = "sha256:0dc64ee3f33cd7899f79a8d788abfbec168410be356ed9bd30bbd3f0a23a7204"}, + {file = "regex-2020.7.14-cp38-cp38-manylinux1_i686.whl", hash = "sha256:5ea81ea3dbd6767873c611687141ec7b06ed8bab43f68fad5b7be184a920dc99"}, + {file = "regex-2020.7.14-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:bbb332d45b32df41200380fff14712cb6093b61bd142272a10b16778c418e98e"}, + {file = "regex-2020.7.14-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:c11d6033115dc4887c456565303f540c44197f4fc1a2bfb192224a301534888e"}, + {file = "regex-2020.7.14-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:75aaa27aa521a182824d89e5ab0a1d16ca207318a6b65042b046053cfc8ed07a"}, + {file = "regex-2020.7.14-cp38-cp38-win32.whl", hash = "sha256:d6cff2276e502b86a25fd10c2a96973fdb45c7a977dca2138d661417f3728341"}, + {file = "regex-2020.7.14-cp38-cp38-win_amd64.whl", hash = "sha256:7a2dd66d2d4df34fa82c9dc85657c5e019b87932019947faece7983f2089a840"}, + {file = "regex-2020.7.14.tar.gz", hash = "sha256:3a3af27a8d23143c49a3420efe5b3f8cf1a48c6fc8bc6856b03f638abc1833bb"}, +] six = [ {file = "six-1.15.0-py2.py3-none-any.whl", hash = "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"}, {file = "six-1.15.0.tar.gz", hash = "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259"}, ] +toml = [ + {file = "toml-0.10.1-py2.py3-none-any.whl", hash = "sha256:bda89d5935c2eac546d648028b9901107a595863cb36bae0c73ac804a9b4ce88"}, + {file = "toml-0.10.1.tar.gz", hash = "sha256:926b612be1e5ce0634a2ca03470f95169cf16f939018233a670519cb4ac58b0f"}, +] +typed-ast = [ + {file = "typed_ast-1.4.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:73d785a950fc82dd2a25897d525d003f6378d1cb23ab305578394694202a58c3"}, + {file = "typed_ast-1.4.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:aaee9905aee35ba5905cfb3c62f3e83b3bec7b39413f0a7f19be4e547ea01ebb"}, + {file = "typed_ast-1.4.1-cp35-cp35m-win32.whl", hash = "sha256:0c2c07682d61a629b68433afb159376e24e5b2fd4641d35424e462169c0a7919"}, + {file = "typed_ast-1.4.1-cp35-cp35m-win_amd64.whl", hash = "sha256:4083861b0aa07990b619bd7ddc365eb7fa4b817e99cf5f8d9cf21a42780f6e01"}, + {file = "typed_ast-1.4.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:269151951236b0f9a6f04015a9004084a5ab0d5f19b57de779f908621e7d8b75"}, + {file = "typed_ast-1.4.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:24995c843eb0ad11a4527b026b4dde3da70e1f2d8806c99b7b4a7cf491612652"}, + {file = "typed_ast-1.4.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:fe460b922ec15dd205595c9b5b99e2f056fd98ae8f9f56b888e7a17dc2b757e7"}, + {file = "typed_ast-1.4.1-cp36-cp36m-win32.whl", hash = "sha256:4e3e5da80ccbebfff202a67bf900d081906c358ccc3d5e3c8aea42fdfdfd51c1"}, + {file = "typed_ast-1.4.1-cp36-cp36m-win_amd64.whl", hash = "sha256:249862707802d40f7f29f6e1aad8d84b5aa9e44552d2cc17384b209f091276aa"}, + {file = "typed_ast-1.4.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:8ce678dbaf790dbdb3eba24056d5364fb45944f33553dd5869b7580cdbb83614"}, + {file = "typed_ast-1.4.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:c9e348e02e4d2b4a8b2eedb48210430658df6951fa484e59de33ff773fbd4b41"}, + {file = "typed_ast-1.4.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:bcd3b13b56ea479b3650b82cabd6b5343a625b0ced5429e4ccad28a8973f301b"}, + {file = "typed_ast-1.4.1-cp37-cp37m-win32.whl", hash = "sha256:d5d33e9e7af3b34a40dc05f498939f0ebf187f07c385fd58d591c533ad8562fe"}, + {file = "typed_ast-1.4.1-cp37-cp37m-win_amd64.whl", hash = "sha256:0666aa36131496aed8f7be0410ff974562ab7eeac11ef351def9ea6fa28f6355"}, + {file = "typed_ast-1.4.1-cp38-cp38-macosx_10_15_x86_64.whl", hash = "sha256:d205b1b46085271b4e15f670058ce182bd1199e56b317bf2ec004b6a44f911f6"}, + {file = "typed_ast-1.4.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:6daac9731f172c2a22ade6ed0c00197ee7cc1221aa84cfdf9c31defeb059a907"}, + {file = "typed_ast-1.4.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:498b0f36cc7054c1fead3d7fc59d2150f4d5c6c56ba7fb150c013fbc683a8d2d"}, + {file = "typed_ast-1.4.1-cp38-cp38-win32.whl", hash = "sha256:715ff2f2df46121071622063fc7543d9b1fd19ebfc4f5c8895af64a77a8c852c"}, + {file = "typed_ast-1.4.1-cp38-cp38-win_amd64.whl", hash = "sha256:fc0fea399acb12edbf8a628ba8d2312f583bdbdb3335635db062fa98cf71fca4"}, + {file = "typed_ast-1.4.1-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:d43943ef777f9a1c42bf4e552ba23ac77a6351de620aa9acf64ad54933ad4d34"}, + {file = "typed_ast-1.4.1.tar.gz", hash = "sha256:8c8aaad94455178e3187ab22c8b01a3837f8ee50e09cf31f1ba129eb293ec30b"}, +] virtualenv = [ {file = "virtualenv-20.0.29-py2.py3-none-any.whl", hash = "sha256:8aa9c37b082664dbce2236fa420759c02d64109d8e6013593ad13914718a30fd"}, {file = "virtualenv-20.0.29.tar.gz", hash = "sha256:f14a0a98ea4397f0d926cff950361766b6a73cd5975ae7eb259d12919f819a25"}, diff --git a/pyproject.toml b/pyproject.toml index 158fb5d..8fe4449 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -2,6 +2,11 @@ build-backend = "poetry.masonry.api" requires = ["poetry>=0.12"] +[tool.black] +line-length = 88 +skip-string-normalization = true +target-version = ["py38"] + [tool.poetry] name = "urban-meal-delivery" version = "0.1.0.dev0" @@ -25,3 +30,8 @@ python = "^3.8" [tool.poetry.dev-dependencies] # Task Runners nox = "^2020.5.24" + +# Code Formatters +autoflake = "^1.3.1" +black = "^19.10b0" +isort = "^5.2.2" diff --git a/setup.cfg b/setup.cfg new file mode 100644 index 0000000..534a0f5 --- /dev/null +++ b/setup.cfg @@ -0,0 +1,26 @@ +[black] +# black's settings are in pyproject.toml => [tool.black] + + +[isort] +atomic = true +case_sensitive = true +combine_star = true +force_alphabetical_sort_within_sections = true +lines_after_imports = 2 +remove_redundant_aliases = true + +# Comply with black's style. +# Source: https://github.com/psf/black/blob/master/docs/compatible_configs.md#isort +ensure_newline_before_comments = true +force_grid_wrap = 0 +include_trailing_comma = true +line_length = 88 +multi_line_output = 3 +use_parentheses = true + +# Comply with Google's Python Style Guide. +# All imports go on a single line except the ones from the typing module. +# Source: https://google.github.io/styleguide/pyguide.html#313-imports-formatting +force_single_line = true +single_line_exclusions = typing From c7989e0040b1e98ad0e6bc0d9ba70ee3a5e98b86 Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Mon, 3 Aug 2020 22:41:37 +0200 Subject: [PATCH 03/13] Add a code linting tool chain - use flake8 as the main and pylint as the auxiliary linter - install flake8 with the following plug-ins: + flake8-annotations => enforce type annotations for functions/classes + flake8-black => ensure black would not make any changes + flake8-expression-complexity + wemake-python-styleguide, which packages the following: * darglint * flake8-bandit * flake8-broken-line * flake8-bugbear * flake8-commas * flake8-comprehensions * flake8-debugger * flake8-docstrings * flake8-eradicate * flake8-isort * flake8-rst-docstrings * flake8-string-format * flake8-quotes * pep8-naming - configure flake8 & friends in a rather explicit and strict way - isort needed to be downgraded to ^4.3.21 due to a conflict with pylint and wemake-python-styleguide: + provide TODO's to remove the parts that "fix" isort - use mypy for static type checking - add a nox session "lint" that runs flake8, mypy, and pylint - lint all source files --- noxfile.py | 67 ++++- poetry.lock | 738 ++++++++++++++++++++++++++++++++++++++++++++++++- pyproject.toml | 13 +- setup.cfg | 144 ++++++++++ 4 files changed, 938 insertions(+), 24 deletions(-) diff --git a/noxfile.py b/noxfile.py index 2a72dfe..683ab79 100644 --- a/noxfile.py +++ b/noxfile.py @@ -1,8 +1,9 @@ """Configure nox for the isolated test and lint environments.""" +import contextlib import os import tempfile -from typing import Any +from typing import Any, Generator import nox from nox.sessions import Session @@ -34,7 +35,7 @@ nox.options.sessions = ( @nox.session(name='format', python=MAIN_PYTHON) -def format_(session: Session): +def format_(session: Session) -> None: """Format source files with autoflake, black, and isort.""" _begin(session) _install_packages(session, 'autoflake', 'black', 'isort') @@ -54,24 +55,52 @@ def format_(session: Session): ) session.run('black', '--version') session.run('black', *locations) - session.run('isort', '--version') - session.run('isort', *locations) + with _isort_fix(session): # TODO (isort): Remove after upgrading + session.run('isort', '--version') + session.run('isort', *locations) @nox.session(python=MAIN_PYTHON) -def lint(session: Session): - """Lint source files.""" +def lint(session: Session) -> None: + """Lint source files with flake8, mypy, and pylint.""" _begin(session) + _install_packages( + session, + 'flake8', + 'flake8-annotations', + 'flake8-black', + 'flake8-expression-complexity', + 'mypy', + 'pylint', + 'wemake-python-styleguide', + ) + # Interpret extra arguments as locations of source files. + locations = session.posargs or SRC_LOCATIONS + session.run('flake8', '--version') + session.run('flake8', '--ignore=I0', *locations) # TODO (isort): Remove flag + with _isort_fix(session): # TODO (isort): Remove after upgrading + session.run('isort', '--version') + session.run('isort', '--check-only', *locations) + session.run('mypy', '--version') + session.run('mypy', *locations) + # Ignore errors where pylint cannot import a third-party package due its + # being run in an isolated environment. One way to fix this is to install + # all develop dependencies in this nox session, which we do not do. The + # whole point of static linting tools is to not rely on any package be + # importable at runtime. Instead, these imports are validated implicitly + # when the test suite is run. + session.run('pylint', '--version') + session.run('pylint', '--disable=import-error', *locations) @nox.session(python=[MAIN_PYTHON, NEXT_PYTHON]) -def test(session: Session): +def test(session: Session) -> None: """Test the code base.""" _begin(session) @nox.session(name='pre-commit', python=MAIN_PYTHON, venv_backend='none') -def pre_commit(session: Session): +def pre_commit(session: Session) -> None: """Source files must be well-formed before they enter git.""" _begin(session) session.notify('format') @@ -79,7 +108,7 @@ def pre_commit(session: Session): @nox.session(name='pre-merge', python=MAIN_PYTHON, venv_backend='none') -def pre_merge(session: Session): +def pre_merge(session: Session) -> None: """The test suite must pass before merges are made.""" _begin(session) session.notify('test') @@ -88,17 +117,17 @@ def pre_merge(session: Session): def _begin(session: Session) -> None: """Show generic info about a session.""" if session.posargs: - print('extra arguments:', *session.posargs) + print('extra arguments:', *session.posargs) # noqa:WPS421 session.run('python', '--version') # Fake GNU's pwd. session.log('pwd') - print(os.getcwd()) + print(os.getcwd()) # noqa:WPS421 def _install_packages( - session: Session, *packages_or_pip_args: str, **kwargs: Any + session: Session, *packages_or_pip_args: str, **kwargs: Any, ) -> None: """Install packages respecting the poetry.lock file. @@ -110,7 +139,7 @@ def _install_packages( session: the Session object *packages_or_pip_args: the packages to be installed or pip options **kwargs: passed on to nox.sessions.Session.install() - """ + """ # noqa:RST210,RST213 if session.virtualenv.reuse_existing: session.log( 'No dependencies are installed as an existing environment is re-used', @@ -131,3 +160,15 @@ def _install_packages( session.install( f'--constraint={requirements_txt.name}', *packages_or_pip_args, **kwargs, ) + + +# TODO (isort): Remove this fix after +# upgrading to isort ^5.2.2 in pyproject.toml. +@contextlib.contextmanager +def _isort_fix(session: Session) -> Generator: + """Temporarily upgrade to isort 5.2.2.""" + session.install('isort==5.2.2') + try: + yield + finally: + session.install('isort==4.3.21') diff --git a/poetry.lock b/poetry.lock index ffceece..2c6ed11 100644 --- a/poetry.lock +++ b/poetry.lock @@ -17,6 +17,38 @@ version = "1.12.0" [package.extras] test = ["coverage", "flake8", "pexpect", "wheel"] +[[package]] +category = "dev" +description = "Read/rewrite/write Python ASTs" +name = "astor" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,>=2.7" +version = "0.8.1" + +[[package]] +category = "dev" +description = "Pretty print the output of python stdlib `ast.parse`." +name = "astpretty" +optional = false +python-versions = ">=3.6.1" +version = "2.0.0" + +[package.extras] +typed = ["typed-ast"] + +[[package]] +category = "dev" +description = "An abstract syntax tree for Python with inference support." +name = "astroid" +optional = false +python-versions = ">=3.5" +version = "2.4.2" + +[package.dependencies] +lazy-object-proxy = ">=1.4.0,<1.5.0" +six = ">=1.12,<2.0" +wrapt = ">=1.11,<2.0" + [[package]] category = "dev" description = "Classes Without Boilerplate" @@ -42,6 +74,21 @@ version = "1.3.1" [package.dependencies] pyflakes = ">=1.1.0" +[[package]] +category = "dev" +description = "Security oriented static analyser for python code." +name = "bandit" +optional = false +python-versions = "*" +version = "1.6.2" + +[package.dependencies] +GitPython = ">=1.0.1" +PyYAML = ">=3.13" +colorama = ">=0.3.9" +six = ">=1.10.0" +stevedore = ">=1.20.0" + [[package]] category = "dev" description = "The uncompromising code formatter." @@ -73,7 +120,7 @@ version = "7.1.2" [[package]] category = "dev" description = "Cross-platform colored terminal text." -marker = "sys_platform == \"win32\"" +marker = "sys_platform == \"win32\" or platform_system == \"Windows\"" name = "colorama" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" @@ -90,6 +137,14 @@ version = "4.2.1" [package.dependencies] colorama = "*" +[[package]] +category = "dev" +description = "A utility for ensuring Google-style docstrings stay up to date with the source code." +name = "darglint" +optional = false +python-versions = ">=3.5,<4.0" +version = "1.5.2" + [[package]] category = "dev" description = "Distribution utilities" @@ -98,6 +153,22 @@ optional = false python-versions = "*" version = "0.3.1" +[[package]] +category = "dev" +description = "Docutils -- Python Documentation Utilities" +name = "docutils" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +version = "0.16" + +[[package]] +category = "dev" +description = "Removes commented-out code." +name = "eradicate" +optional = false +python-versions = "*" +version = "1.0" + [[package]] category = "dev" description = "A platform independent file lock." @@ -106,18 +177,290 @@ optional = false python-versions = "*" version = "3.0.12" +[[package]] +category = "dev" +description = "the modular source code checker: pep8 pyflakes and co" +name = "flake8" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,>=2.7" +version = "3.8.3" + +[package.dependencies] +mccabe = ">=0.6.0,<0.7.0" +pycodestyle = ">=2.6.0a1,<2.7.0" +pyflakes = ">=2.2.0,<2.3.0" + +[[package]] +category = "dev" +description = "Flake8 Type Annotation Checks" +name = "flake8-annotations" +optional = false +python-versions = ">=3.6.1,<4.0.0" +version = "2.3.0" + +[package.dependencies] +flake8 = ">=3.7,<3.9" + +[[package]] +category = "dev" +description = "Automated security testing with bandit and flake8." +name = "flake8-bandit" +optional = false +python-versions = "*" +version = "2.1.2" + +[package.dependencies] +bandit = "*" +flake8 = "*" +flake8-polyfill = "*" +pycodestyle = "*" + +[[package]] +category = "dev" +description = "flake8 plugin to call black as a code style validator" +name = "flake8-black" +optional = false +python-versions = "*" +version = "0.2.1" + +[package.dependencies] +black = "*" +flake8 = ">=3.0.0" + +[[package]] +category = "dev" +description = "Flake8 plugin to forbid backslashes for line breaks" +name = "flake8-broken-line" +optional = false +python-versions = ">=3.6,<4.0" +version = "0.2.1" + +[package.dependencies] +flake8 = ">=3.5,<4.0" + +[[package]] +category = "dev" +description = "A plugin for flake8 finding likely bugs and design problems in your program. Contains warnings that don't belong in pyflakes and pycodestyle." +name = "flake8-bugbear" +optional = false +python-versions = ">=3.5" +version = "19.8.0" + +[package.dependencies] +attrs = "*" +flake8 = ">=3.0.0" + +[[package]] +category = "dev" +description = "Flake8 lint for trailing commas." +name = "flake8-commas" +optional = false +python-versions = "*" +version = "2.0.0" + +[package.dependencies] +flake8 = ">=2,<4.0.0" + +[[package]] +category = "dev" +description = "A flake8 plugin to help you write better list/set/dict comprehensions." +name = "flake8-comprehensions" +optional = false +python-versions = ">=3.5" +version = "3.2.3" + +[package.dependencies] +flake8 = ">=3.0,<3.2.0 || >3.2.0,<4" + +[[package]] +category = "dev" +description = "ipdb/pdb statement checker plugin for flake8" +name = "flake8-debugger" +optional = false +python-versions = "*" +version = "3.2.1" + +[package.dependencies] +flake8 = ">=1.5" +pycodestyle = "*" + +[[package]] +category = "dev" +description = "Extension for flake8 which uses pydocstyle to check docstrings" +name = "flake8-docstrings" +optional = false +python-versions = "*" +version = "1.5.0" + +[package.dependencies] +flake8 = ">=3" +pydocstyle = ">=2.1" + +[[package]] +category = "dev" +description = "Flake8 plugin to find commented out code" +name = "flake8-eradicate" +optional = false +python-versions = ">=3.6,<4.0" +version = "0.3.0" + +[package.dependencies] +attrs = "*" +eradicate = ">=1.0,<2.0" +flake8 = ">=3.5,<4.0" + +[[package]] +category = "dev" +description = "A flake8 extension that checks expressions complexity" +name = "flake8-expression-complexity" +optional = false +python-versions = ">=3.6" +version = "0.0.8" + +[package.dependencies] +astpretty = "*" +flake8 = "*" +setuptools = "*" + +[[package]] +category = "dev" +description = "flake8 plugin that integrates isort ." +name = "flake8-isort" +optional = false +python-versions = "*" +version = "3.0.1" + +[package.dependencies] +flake8 = ">=3.2.1,<4" +testfixtures = ">=6.8.0,<7" + +[package.dependencies.isort] +extras = ["pyproject"] +version = ">=4.3.5,<5" + +[package.extras] +test = ["pytest (>=4.0.2,<6)"] + +[[package]] +category = "dev" +description = "Polyfill package for Flake8 plugins" +name = "flake8-polyfill" +optional = false +python-versions = "*" +version = "1.0.2" + +[package.dependencies] +flake8 = "*" + +[[package]] +category = "dev" +description = "Flake8 lint for quotes." +name = "flake8-quotes" +optional = false +python-versions = "*" +version = "2.1.2" + +[package.dependencies] +flake8 = "*" + +[[package]] +category = "dev" +description = "Python docstring reStructuredText (RST) validator" +name = "flake8-rst-docstrings" +optional = false +python-versions = "*" +version = "0.0.12" + +[package.dependencies] +flake8 = ">=3.0.0" +restructuredtext_lint = "*" + +[[package]] +category = "dev" +description = "string format checker, plugin for flake8" +name = "flake8-string-format" +optional = false +python-versions = "*" +version = "0.2.3" + +[package.dependencies] +flake8 = "*" + +[[package]] +category = "dev" +description = "Git Object Database" +name = "gitdb" +optional = false +python-versions = ">=3.4" +version = "4.0.5" + +[package.dependencies] +smmap = ">=3.0.1,<4" + +[[package]] +category = "dev" +description = "Python Git Library" +name = "gitpython" +optional = false +python-versions = ">=3.4" +version = "3.1.7" + +[package.dependencies] +gitdb = ">=4.0.1,<5" + [[package]] category = "dev" description = "A Python utility / library to sort Python imports." name = "isort" optional = false -python-versions = ">=3.6,<4.0" -version = "5.2.2" +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "4.3.21" [package.extras] -colors = ["colorama (>=0.4.3,<0.5.0)"] -pipfile_deprecated_finder = ["pipreqs", "requirementslib", "tomlkit (>=0.5.3)"] -requirements_deprecated_finder = ["pipreqs", "pip-api"] +pipfile = ["pipreqs", "requirementslib"] +pyproject = ["toml"] +requirements = ["pipreqs", "pip-api"] +xdg_home = ["appdirs (>=1.4.0)"] + +[[package]] +category = "dev" +description = "A fast and thorough lazy object proxy." +name = "lazy-object-proxy" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "1.4.3" + +[[package]] +category = "dev" +description = "McCabe checker, plugin for flake8" +name = "mccabe" +optional = false +python-versions = "*" +version = "0.6.1" + +[[package]] +category = "dev" +description = "Optional static typing for Python" +name = "mypy" +optional = false +python-versions = ">=3.5" +version = "0.782" + +[package.dependencies] +mypy-extensions = ">=0.4.3,<0.5.0" +typed-ast = ">=1.4.0,<1.5.0" +typing-extensions = ">=3.7.4" + +[package.extras] +dmypy = ["psutil (>=4.0)"] + +[[package]] +category = "dev" +description = "Experimental type system extensions for programs checked with the mypy typechecker." +name = "mypy-extensions" +optional = false +python-versions = "*" +version = "0.4.3" [[package]] category = "dev" @@ -144,6 +487,25 @@ optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" version = "0.8.0" +[[package]] +category = "dev" +description = "Python Build Reasonableness" +name = "pbr" +optional = false +python-versions = "*" +version = "5.4.5" + +[[package]] +category = "dev" +description = "Check PEP-8 naming conventions, plugin for flake8" +name = "pep8-naming" +optional = false +python-versions = "*" +version = "0.9.1" + +[package.dependencies] +flake8-polyfill = ">=1.0.2,<2" + [[package]] category = "dev" description = "library with cross-python path, ini-parsing, io, code, log facilities" @@ -152,6 +514,25 @@ optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" version = "1.9.0" +[[package]] +category = "dev" +description = "Python style guide checker" +name = "pycodestyle" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "2.6.0" + +[[package]] +category = "dev" +description = "Python docstring style checker" +name = "pydocstyle" +optional = false +python-versions = ">=3.5" +version = "5.0.2" + +[package.dependencies] +snowballstemmer = "*" + [[package]] category = "dev" description = "passive checker of Python programs" @@ -160,6 +541,37 @@ optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" version = "2.2.0" +[[package]] +category = "dev" +description = "Pygments is a syntax highlighting package written in Python." +name = "pygments" +optional = false +python-versions = ">=3.5" +version = "2.6.1" + +[[package]] +category = "dev" +description = "python code static checker" +name = "pylint" +optional = false +python-versions = ">=3.5.*" +version = "2.5.3" + +[package.dependencies] +astroid = ">=2.4.0,<=2.5" +colorama = "*" +isort = ">=4.2.5,<5" +mccabe = ">=0.6,<0.7" +toml = ">=0.7.1" + +[[package]] +category = "dev" +description = "YAML parser and emitter for Python" +name = "pyyaml" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +version = "5.3.1" + [[package]] category = "dev" description = "Alternative regular expression module, to replace re." @@ -168,6 +580,17 @@ optional = false python-versions = "*" version = "2020.7.14" +[[package]] +category = "dev" +description = "reStructuredText linter" +name = "restructuredtext-lint" +optional = false +python-versions = "*" +version = "1.3.1" + +[package.dependencies] +docutils = ">=0.11,<1.0" + [[package]] category = "dev" description = "Python 2 and 3 compatibility utilities" @@ -176,6 +599,46 @@ optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" version = "1.15.0" +[[package]] +category = "dev" +description = "A pure Python implementation of a sliding window memory map manager" +name = "smmap" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "3.0.4" + +[[package]] +category = "dev" +description = "This package provides 26 stemmers for 25 languages generated from Snowball algorithms." +name = "snowballstemmer" +optional = false +python-versions = "*" +version = "2.0.0" + +[[package]] +category = "dev" +description = "Manage dynamic plugins for Python applications" +name = "stevedore" +optional = false +python-versions = ">=3.6" +version = "3.2.0" + +[package.dependencies] +pbr = ">=2.0.0,<2.1.0 || >2.1.0" + +[[package]] +category = "dev" +description = "A collection of helpers and mock objects for unit tests and doc tests." +name = "testfixtures" +optional = false +python-versions = "*" +version = "6.14.1" + +[package.extras] +build = ["setuptools-git", "wheel", "twine"] +docs = ["sphinx", "zope.component", "sybil", "twisted", "mock", "django (<2)", "django"] +test = ["pytest (>=3.6)", "pytest-cov", "pytest-django", "zope.component", "sybil", "twisted", "mock", "django (<2)", "django"] + [[package]] category = "dev" description = "Python Library for Tom's Obvious, Minimal Language" @@ -192,6 +655,14 @@ optional = false python-versions = "*" version = "1.4.1" +[[package]] +category = "dev" +description = "Backported and Experimental Type Hints for Python 3.5+" +name = "typing-extensions" +optional = false +python-versions = "*" +version = "3.7.4.2" + [[package]] category = "dev" description = "Virtual Python Environment builder" @@ -210,8 +681,45 @@ six = ">=1.9.0,<2" docs = ["proselint (>=0.10.2)", "sphinx (>=3)", "sphinx-argparse (>=0.2.5)", "sphinx-rtd-theme (>=0.4.3)", "towncrier (>=19.9.0rc1)"] testing = ["coverage (>=5)", "coverage-enable-subprocess (>=1)", "flaky (>=3)", "pytest (>=4)", "pytest-env (>=0.6.2)", "pytest-freezegun (>=0.4.1)", "pytest-mock (>=2)", "pytest-randomly (>=1)", "pytest-timeout (>=1)", "pytest-xdist (>=1.31.0)", "packaging (>=20.0)", "xonsh (>=0.9.16)"] +[[package]] +category = "dev" +description = "The strictest and most opinionated python linter ever" +name = "wemake-python-styleguide" +optional = false +python-versions = ">=3.6,<4.0" +version = "0.14.1" + +[package.dependencies] +astor = ">=0.8,<0.9" +attrs = "*" +darglint = ">=1.2,<2.0" +flake8 = ">=3.7,<4.0" +flake8-bandit = ">=2.1,<3.0" +flake8-broken-line = ">=0.2,<0.3" +flake8-bugbear = ">=19.3,<20.0" +flake8-commas = ">=2.0,<3.0" +flake8-comprehensions = ">=3.1.0,<4.0.0" +flake8-debugger = ">=3.1,<4.0" +flake8-docstrings = ">=1.3.1,<2.0.0" +flake8-eradicate = ">=0.3,<0.4" +flake8-isort = ">=3.0.1,<4" +flake8-quotes = ">=2.0.1,<3.0.0" +flake8-rst-docstrings = ">=0.0.12,<0.0.13" +flake8-string-format = ">=0.2,<0.3" +pep8-naming = ">=0.9.1,<0.10.0" +pygments = ">=2.4,<3.0" +typing_extensions = ">=3.6,<4.0" + +[[package]] +category = "dev" +description = "Module for decorators, wrappers and monkey patching." +name = "wrapt" +optional = false +python-versions = "*" +version = "1.12.1" + [metadata] -content-hash = "9894f7290cbab39b1fa0f3d0ba9cd2aebe16228169867415a69e039ee2b12425" +content-hash = "e69534c9133ef3fb975f33937c97711f69281a8ffb931097a7b9e4ae1132fa09" lock-version = "1.0" python-versions = "^3.8" @@ -224,6 +732,18 @@ argcomplete = [ {file = "argcomplete-1.12.0-py2.py3-none-any.whl", hash = "sha256:91dc7f9c7f6281d5a0dce5e73d2e33283aaef083495c13974a7dd197a1cdc949"}, {file = "argcomplete-1.12.0.tar.gz", hash = "sha256:2fbe5ed09fd2c1d727d4199feca96569a5b50d44c71b16da9c742201f7cc295c"}, ] +astor = [ + {file = "astor-0.8.1-py2.py3-none-any.whl", hash = "sha256:070a54e890cefb5b3739d19f30f5a5ec840ffc9c50ffa7d23cc9fc1a38ebbfc5"}, + {file = "astor-0.8.1.tar.gz", hash = "sha256:6a6effda93f4e1ce9f618779b2dd1d9d84f1e32812c23a29b3fff6fd7f63fa5e"}, +] +astpretty = [ + {file = "astpretty-2.0.0-py2.py3-none-any.whl", hash = "sha256:7f27633ed885033da8b58666e7079ffff7e8e01869ec1aa66484cb5185ea3aa4"}, + {file = "astpretty-2.0.0.tar.gz", hash = "sha256:e4724bfd753636ba4a84384702e9796e5356969f40af2596d846ce64addde086"}, +] +astroid = [ + {file = "astroid-2.4.2-py3-none-any.whl", hash = "sha256:bc58d83eb610252fd8de6363e39d4f1d0619c894b0ed24603b881c02e64c7386"}, + {file = "astroid-2.4.2.tar.gz", hash = "sha256:2f4078c2a41bf377eea06d71c9d2ba4eb8f6b1af2135bec27bbbb7d8f12bb703"}, +] attrs = [ {file = "attrs-19.3.0-py2.py3-none-any.whl", hash = "sha256:08a96c641c3a74e44eb59afb61a24f2cb9f4d7188748e76ba4bb5edfa3cb7d1c"}, {file = "attrs-19.3.0.tar.gz", hash = "sha256:f7b7ce16570fe9965acd6d30101a28f62fb4a7f9e926b3bbc9b61f8b04247e72"}, @@ -231,6 +751,10 @@ attrs = [ autoflake = [ {file = "autoflake-1.3.1.tar.gz", hash = "sha256:680cb9dade101ed647488238ccb8b8bfb4369b53d58ba2c8cdf7d5d54e01f95b"}, ] +bandit = [ + {file = "bandit-1.6.2-py2.py3-none-any.whl", hash = "sha256:336620e220cf2d3115877685e264477ff9d9abaeb0afe3dc7264f55fa17a3952"}, + {file = "bandit-1.6.2.tar.gz", hash = "sha256:41e75315853507aa145d62a78a2a6c5e3240fe14ee7c601459d0df9418196065"}, +] black = [ {file = "black-19.10b0-py36-none-any.whl", hash = "sha256:1b30e59be925fafc1ee4565e5e08abef6b03fe455102883820fe5ee2e4734e0b"}, {file = "black-19.10b0.tar.gz", hash = "sha256:c2edb73a08e9e0e6f65a0e6af18b059b8b1cdd5bef997d7a0b181df93dc81539"}, @@ -247,17 +771,145 @@ colorlog = [ {file = "colorlog-4.2.1-py2.py3-none-any.whl", hash = "sha256:43597fd822ce705190fc997519342fdaaf44b9b47f896ece7aa153ed4b909c74"}, {file = "colorlog-4.2.1.tar.gz", hash = "sha256:75e55822c3a3387d721579241e776de2cf089c9ef9528b1f09e8b04d403ad118"}, ] +darglint = [ + {file = "darglint-1.5.2-py3-none-any.whl", hash = "sha256:049a98cf3aec8cf6ea344a863c68112d80b7f8de214459b5fa6853371f89c3e7"}, + {file = "darglint-1.5.2.tar.gz", hash = "sha256:6b9461f96694c2cf1d8edb1597a783fe6840953b0eb18cc6cc1e72a26f196d79"}, +] distlib = [ {file = "distlib-0.3.1-py2.py3-none-any.whl", hash = "sha256:8c09de2c67b3e7deef7184574fc060ab8a793e7adbb183d942c389c8b13c52fb"}, {file = "distlib-0.3.1.zip", hash = "sha256:edf6116872c863e1aa9d5bb7cb5e05a022c519a4594dc703843343a9ddd9bff1"}, ] +docutils = [ + {file = "docutils-0.16-py2.py3-none-any.whl", hash = "sha256:0c5b78adfbf7762415433f5515cd5c9e762339e23369dbe8000d84a4bf4ab3af"}, + {file = "docutils-0.16.tar.gz", hash = "sha256:c2de3a60e9e7d07be26b7f2b00ca0309c207e06c100f9cc2a94931fc75a478fc"}, +] +eradicate = [ + {file = "eradicate-1.0.tar.gz", hash = "sha256:4ffda82aae6fd49dfffa777a857cb758d77502a1f2e0f54c9ac5155a39d2d01a"}, +] filelock = [ {file = "filelock-3.0.12-py3-none-any.whl", hash = "sha256:929b7d63ec5b7d6b71b0fa5ac14e030b3f70b75747cef1b10da9b879fef15836"}, {file = "filelock-3.0.12.tar.gz", hash = "sha256:18d82244ee114f543149c66a6e0c14e9c4f8a1044b5cdaadd0f82159d6a6ff59"}, ] +flake8 = [ + {file = "flake8-3.8.3-py2.py3-none-any.whl", hash = "sha256:15e351d19611c887e482fb960eae4d44845013cc142d42896e9862f775d8cf5c"}, + {file = "flake8-3.8.3.tar.gz", hash = "sha256:f04b9fcbac03b0a3e58c0ab3a0ecc462e023a9faf046d57794184028123aa208"}, +] +flake8-annotations = [ + {file = "flake8-annotations-2.3.0.tar.gz", hash = "sha256:8d18db74a750dd97f40b483cc3ef80d07d03f687525bad8fd83365dcd3bfd414"}, + {file = "flake8_annotations-2.3.0-py3-none-any.whl", hash = "sha256:7816a5d8f65ffdf37b8e21e5b17e0fd1e492aa92638573276de066e889a22b26"}, +] +flake8-bandit = [ + {file = "flake8_bandit-2.1.2.tar.gz", hash = "sha256:687fc8da2e4a239b206af2e54a90093572a60d0954f3054e23690739b0b0de3b"}, +] +flake8-black = [ + {file = "flake8-black-0.2.1.tar.gz", hash = "sha256:f26651bc10db786c03f4093414f7c9ea982ed8a244cec323c984feeffdf4c118"}, +] +flake8-broken-line = [ + {file = "flake8-broken-line-0.2.1.tar.gz", hash = "sha256:414477070231a5aa05468d48db2742a594b53fbc1ecba28044646706a11fb861"}, + {file = "flake8_broken_line-0.2.1-py3-none-any.whl", hash = "sha256:75858359e3ccd4f1d92a9e7582aa5c9e4485cbc920dd05954703900cf907667e"}, +] +flake8-bugbear = [ + {file = "flake8-bugbear-19.8.0.tar.gz", hash = "sha256:d8c466ea79d5020cb20bf9f11cf349026e09517a42264f313d3f6fddb83e0571"}, + {file = "flake8_bugbear-19.8.0-py35.py36.py37-none-any.whl", hash = "sha256:ded4d282778969b5ab5530ceba7aa1a9f1b86fa7618fc96a19a1d512331640f8"}, +] +flake8-commas = [ + {file = "flake8-commas-2.0.0.tar.gz", hash = "sha256:d3005899466f51380387df7151fb59afec666a0f4f4a2c6a8995b975de0f44b7"}, + {file = "flake8_commas-2.0.0-py2.py3-none-any.whl", hash = "sha256:ee2141a3495ef9789a3894ed8802d03eff1eaaf98ce6d8653a7c573ef101935e"}, +] +flake8-comprehensions = [ + {file = "flake8-comprehensions-3.2.3.tar.gz", hash = "sha256:d5751acc0f7364794c71d06f113f4686d6e2e26146a50fa93130b9f200fe160d"}, + {file = "flake8_comprehensions-3.2.3-py3-none-any.whl", hash = "sha256:44eaae9894aa15f86e0c86df1e218e7917494fab6f96d28f96a029c460f17d92"}, +] +flake8-debugger = [ + {file = "flake8-debugger-3.2.1.tar.gz", hash = "sha256:712d7c1ff69ddf3f0130e94cc88c2519e720760bce45e8c330bfdcb61ab4090d"}, +] +flake8-docstrings = [ + {file = "flake8-docstrings-1.5.0.tar.gz", hash = "sha256:3d5a31c7ec6b7367ea6506a87ec293b94a0a46c0bce2bb4975b7f1d09b6f3717"}, + {file = "flake8_docstrings-1.5.0-py2.py3-none-any.whl", hash = "sha256:a256ba91bc52307bef1de59e2a009c3cf61c3d0952dbe035d6ff7208940c2edc"}, +] +flake8-eradicate = [ + {file = "flake8-eradicate-0.3.0.tar.gz", hash = "sha256:d0b3d283d85079917acbfe39b9d637385cd82cba3ae3d76c1278c07ddcf0d9b9"}, + {file = "flake8_eradicate-0.3.0-py3-none-any.whl", hash = "sha256:e8b32b32300bfb407fe7ef74667c8d2d3a6a81bdf6f09c14a7bcc82b7b870f8b"}, +] +flake8-expression-complexity = [ + {file = "flake8_expression_complexity-0.0.8.tar.gz", hash = "sha256:c23f8ae677eb13b073be9bafa2cf443a87d3e2594817ba82c1f3e184e2de4afa"}, +] +flake8-isort = [ + {file = "flake8-isort-3.0.1.tar.gz", hash = "sha256:5d976da513cc390232ad5a9bb54aee8a092466a15f442d91dfc525834bee727a"}, + {file = "flake8_isort-3.0.1-py2.py3-none-any.whl", hash = "sha256:df1dd6dd73f6a8b128c9c783356627231783cccc82c13c6dc343d1a5a491699b"}, +] +flake8-polyfill = [ + {file = "flake8-polyfill-1.0.2.tar.gz", hash = "sha256:e44b087597f6da52ec6393a709e7108b2905317d0c0b744cdca6208e670d8eda"}, + {file = "flake8_polyfill-1.0.2-py2.py3-none-any.whl", hash = "sha256:12be6a34ee3ab795b19ca73505e7b55826d5f6ad7230d31b18e106400169b9e9"}, +] +flake8-quotes = [ + {file = "flake8-quotes-2.1.2.tar.gz", hash = "sha256:c844c9592940c8926c60f00bc620808912ff2acd34923ab5338f3a5ca618a331"}, +] +flake8-rst-docstrings = [ + {file = "flake8-rst-docstrings-0.0.12.tar.gz", hash = "sha256:01d38327801781b26c3dfeb71ae37e5a02c5ca1b774a686f63feab8824ca6f9c"}, +] +flake8-string-format = [ + {file = "flake8-string-format-0.2.3.tar.gz", hash = "sha256:774d56103d9242ed968897455ef49b7d6de272000cfa83de5814273a868832f1"}, + {file = "flake8_string_format-0.2.3-py2.py3-none-any.whl", hash = "sha256:68ea72a1a5b75e7018cae44d14f32473c798cf73d75cbaed86c6a9a907b770b2"}, +] +gitdb = [ + {file = "gitdb-4.0.5-py3-none-any.whl", hash = "sha256:91f36bfb1ab7949b3b40e23736db18231bf7593edada2ba5c3a174a7b23657ac"}, + {file = "gitdb-4.0.5.tar.gz", hash = "sha256:c9e1f2d0db7ddb9a704c2a0217be31214e91a4fe1dea1efad19ae42ba0c285c9"}, +] +gitpython = [ + {file = "GitPython-3.1.7-py3-none-any.whl", hash = "sha256:fa3b92da728a457dd75d62bb5f3eb2816d99a7fe6c67398e260637a40e3fafb5"}, + {file = "GitPython-3.1.7.tar.gz", hash = "sha256:2db287d71a284e22e5c2846042d0602465c7434d910406990d5b74df4afb0858"}, +] isort = [ - {file = "isort-5.2.2-py3-none-any.whl", hash = "sha256:aea484023188ef1c38256dd24afa96e914adafe3a911a1786800a74e433006d1"}, - {file = "isort-5.2.2.tar.gz", hash = "sha256:96b27045e3187b9bdde001143b79f9b10a462f372bff7062302818013b6c86f3"}, + {file = "isort-4.3.21-py2.py3-none-any.whl", hash = "sha256:6e811fcb295968434526407adb8796944f1988c5b65e8139058f2014cbe100fd"}, + {file = "isort-4.3.21.tar.gz", hash = "sha256:54da7e92468955c4fceacd0c86bd0ec997b0e1ee80d97f67c35a78b719dccab1"}, +] +lazy-object-proxy = [ + {file = "lazy-object-proxy-1.4.3.tar.gz", hash = "sha256:f3900e8a5de27447acbf900b4750b0ddfd7ec1ea7fbaf11dfa911141bc522af0"}, + {file = "lazy_object_proxy-1.4.3-cp27-cp27m-macosx_10_13_x86_64.whl", hash = "sha256:a2238e9d1bb71a56cd710611a1614d1194dc10a175c1e08d75e1a7bcc250d442"}, + {file = "lazy_object_proxy-1.4.3-cp27-cp27m-win32.whl", hash = "sha256:efa1909120ce98bbb3777e8b6f92237f5d5c8ea6758efea36a473e1d38f7d3e4"}, + {file = "lazy_object_proxy-1.4.3-cp27-cp27m-win_amd64.whl", hash = "sha256:4677f594e474c91da97f489fea5b7daa17b5517190899cf213697e48d3902f5a"}, + {file = "lazy_object_proxy-1.4.3-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:0c4b206227a8097f05c4dbdd323c50edf81f15db3b8dc064d08c62d37e1a504d"}, + {file = "lazy_object_proxy-1.4.3-cp34-cp34m-manylinux1_x86_64.whl", hash = "sha256:d945239a5639b3ff35b70a88c5f2f491913eb94871780ebfabb2568bd58afc5a"}, + {file = "lazy_object_proxy-1.4.3-cp34-cp34m-win32.whl", hash = "sha256:9651375199045a358eb6741df3e02a651e0330be090b3bc79f6d0de31a80ec3e"}, + {file = "lazy_object_proxy-1.4.3-cp34-cp34m-win_amd64.whl", hash = "sha256:eba7011090323c1dadf18b3b689845fd96a61ba0a1dfbd7f24b921398affc357"}, + {file = "lazy_object_proxy-1.4.3-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:48dab84ebd4831077b150572aec802f303117c8cc5c871e182447281ebf3ac50"}, + {file = "lazy_object_proxy-1.4.3-cp35-cp35m-win32.whl", hash = "sha256:ca0a928a3ddbc5725be2dd1cf895ec0a254798915fb3a36af0964a0a4149e3db"}, + {file = "lazy_object_proxy-1.4.3-cp35-cp35m-win_amd64.whl", hash = "sha256:194d092e6f246b906e8f70884e620e459fc54db3259e60cf69a4d66c3fda3449"}, + {file = "lazy_object_proxy-1.4.3-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:97bb5884f6f1cdce0099f86b907aa41c970c3c672ac8b9c8352789e103cf3156"}, + {file = "lazy_object_proxy-1.4.3-cp36-cp36m-win32.whl", hash = "sha256:cb2c7c57005a6804ab66f106ceb8482da55f5314b7fcb06551db1edae4ad1531"}, + {file = "lazy_object_proxy-1.4.3-cp36-cp36m-win_amd64.whl", hash = "sha256:8d859b89baf8ef7f8bc6b00aa20316483d67f0b1cbf422f5b4dc56701c8f2ffb"}, + {file = "lazy_object_proxy-1.4.3-cp37-cp37m-macosx_10_13_x86_64.whl", hash = "sha256:1be7e4c9f96948003609aa6c974ae59830a6baecc5376c25c92d7d697e684c08"}, + {file = "lazy_object_proxy-1.4.3-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:d74bb8693bf9cf75ac3b47a54d716bbb1a92648d5f781fc799347cfc95952383"}, + {file = "lazy_object_proxy-1.4.3-cp37-cp37m-win32.whl", hash = "sha256:9b15f3f4c0f35727d3a0fba4b770b3c4ebbb1fa907dbcc046a1d2799f3edd142"}, + {file = "lazy_object_proxy-1.4.3-cp37-cp37m-win_amd64.whl", hash = "sha256:9254f4358b9b541e3441b007a0ea0764b9d056afdeafc1a5569eee1cc6c1b9ea"}, + {file = "lazy_object_proxy-1.4.3-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:a6ae12d08c0bf9909ce12385803a543bfe99b95fe01e752536a60af2b7797c62"}, + {file = "lazy_object_proxy-1.4.3-cp38-cp38-win32.whl", hash = "sha256:5541cada25cd173702dbd99f8e22434105456314462326f06dba3e180f203dfd"}, + {file = "lazy_object_proxy-1.4.3-cp38-cp38-win_amd64.whl", hash = "sha256:59f79fef100b09564bc2df42ea2d8d21a64fdcda64979c0fa3db7bdaabaf6239"}, +] +mccabe = [ + {file = "mccabe-0.6.1-py2.py3-none-any.whl", hash = "sha256:ab8a6258860da4b6677da4bd2fe5dc2c659cff31b3ee4f7f5d64e79735b80d42"}, + {file = "mccabe-0.6.1.tar.gz", hash = "sha256:dd8d182285a0fe56bace7f45b5e7d1a6ebcbf524e8f3bd87eb0f125271b8831f"}, +] +mypy = [ + {file = "mypy-0.782-cp35-cp35m-macosx_10_6_x86_64.whl", hash = "sha256:2c6cde8aa3426c1682d35190b59b71f661237d74b053822ea3d748e2c9578a7c"}, + {file = "mypy-0.782-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:9c7a9a7ceb2871ba4bac1cf7217a7dd9ccd44c27c2950edbc6dc08530f32ad4e"}, + {file = "mypy-0.782-cp35-cp35m-win_amd64.whl", hash = "sha256:c05b9e4fb1d8a41d41dec8786c94f3b95d3c5f528298d769eb8e73d293abc48d"}, + {file = "mypy-0.782-cp36-cp36m-macosx_10_6_x86_64.whl", hash = "sha256:6731603dfe0ce4352c555c6284c6db0dc935b685e9ce2e4cf220abe1e14386fd"}, + {file = "mypy-0.782-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:f05644db6779387ccdb468cc47a44b4356fc2ffa9287135d05b70a98dc83b89a"}, + {file = "mypy-0.782-cp36-cp36m-win_amd64.whl", hash = "sha256:b7fbfabdbcc78c4f6fc4712544b9b0d6bf171069c6e0e3cb82440dd10ced3406"}, + {file = "mypy-0.782-cp37-cp37m-macosx_10_6_x86_64.whl", hash = "sha256:3fdda71c067d3ddfb21da4b80e2686b71e9e5c72cca65fa216d207a358827f86"}, + {file = "mypy-0.782-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:d7df6eddb6054d21ca4d3c6249cae5578cb4602951fd2b6ee2f5510ffb098707"}, + {file = "mypy-0.782-cp37-cp37m-win_amd64.whl", hash = "sha256:a4a2cbcfc4cbf45cd126f531dedda8485671545b43107ded25ce952aac6fb308"}, + {file = "mypy-0.782-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6bb93479caa6619d21d6e7160c552c1193f6952f0668cdda2f851156e85186fc"}, + {file = "mypy-0.782-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:81c7908b94239c4010e16642c9102bfc958ab14e36048fa77d0be3289dda76ea"}, + {file = "mypy-0.782-cp38-cp38-win_amd64.whl", hash = "sha256:5dd13ff1f2a97f94540fd37a49e5d255950ebcdf446fb597463a40d0df3fac8b"}, + {file = "mypy-0.782-py3-none-any.whl", hash = "sha256:e0b61738ab504e656d1fe4ff0c0601387a5489ca122d55390ade31f9ca0e252d"}, + {file = "mypy-0.782.tar.gz", hash = "sha256:eff7d4a85e9eea55afa34888dfeaccde99e7520b51f867ac28a48492c0b1130c"}, +] +mypy-extensions = [ + {file = "mypy_extensions-0.4.3-py2.py3-none-any.whl", hash = "sha256:090fedd75945a69ae91ce1303b5824f428daf5a028d2f6ab8a299250a846f15d"}, + {file = "mypy_extensions-0.4.3.tar.gz", hash = "sha256:2d82818f5bb3e369420cb3c4060a7970edba416647068eb4c5343488a6c604a8"}, ] nox = [ {file = "nox-2020.5.24-py3-none-any.whl", hash = "sha256:c4509621fead99473a1401870e680b0aadadce5c88440f0532863595176d64c1"}, @@ -267,14 +919,51 @@ pathspec = [ {file = "pathspec-0.8.0-py2.py3-none-any.whl", hash = "sha256:7d91249d21749788d07a2d0f94147accd8f845507400749ea19c1ec9054a12b0"}, {file = "pathspec-0.8.0.tar.gz", hash = "sha256:da45173eb3a6f2a5a487efba21f050af2b41948be6ab52b6a1e3ff22bb8b7061"}, ] +pbr = [ + {file = "pbr-5.4.5-py2.py3-none-any.whl", hash = "sha256:579170e23f8e0c2f24b0de612f71f648eccb79fb1322c814ae6b3c07b5ba23e8"}, + {file = "pbr-5.4.5.tar.gz", hash = "sha256:07f558fece33b05caf857474a366dfcc00562bca13dd8b47b2b3e22d9f9bf55c"}, +] +pep8-naming = [ + {file = "pep8-naming-0.9.1.tar.gz", hash = "sha256:a33d38177056321a167decd6ba70b890856ba5025f0a8eca6a3eda607da93caf"}, + {file = "pep8_naming-0.9.1-py2.py3-none-any.whl", hash = "sha256:45f330db8fcfb0fba57458c77385e288e7a3be1d01e8ea4268263ef677ceea5f"}, +] py = [ {file = "py-1.9.0-py2.py3-none-any.whl", hash = "sha256:366389d1db726cd2fcfc79732e75410e5fe4d31db13692115529d34069a043c2"}, {file = "py-1.9.0.tar.gz", hash = "sha256:9ca6883ce56b4e8da7e79ac18787889fa5206c79dcc67fb065376cd2fe03f342"}, ] +pycodestyle = [ + {file = "pycodestyle-2.6.0-py2.py3-none-any.whl", hash = "sha256:2295e7b2f6b5bd100585ebcb1f616591b652db8a741695b3d8f5d28bdc934367"}, + {file = "pycodestyle-2.6.0.tar.gz", hash = "sha256:c58a7d2815e0e8d7972bf1803331fb0152f867bd89adf8a01dfd55085434192e"}, +] +pydocstyle = [ + {file = "pydocstyle-5.0.2-py3-none-any.whl", hash = "sha256:da7831660b7355307b32778c4a0dbfb137d89254ef31a2b2978f50fc0b4d7586"}, + {file = "pydocstyle-5.0.2.tar.gz", hash = "sha256:f4f5d210610c2d153fae39093d44224c17429e2ad7da12a8b419aba5c2f614b5"}, +] pyflakes = [ {file = "pyflakes-2.2.0-py2.py3-none-any.whl", hash = "sha256:0d94e0e05a19e57a99444b6ddcf9a6eb2e5c68d3ca1e98e90707af8152c90a92"}, {file = "pyflakes-2.2.0.tar.gz", hash = "sha256:35b2d75ee967ea93b55750aa9edbbf72813e06a66ba54438df2cfac9e3c27fc8"}, ] +pygments = [ + {file = "Pygments-2.6.1-py3-none-any.whl", hash = "sha256:ff7a40b4860b727ab48fad6360eb351cc1b33cbf9b15a0f689ca5353e9463324"}, + {file = "Pygments-2.6.1.tar.gz", hash = "sha256:647344a061c249a3b74e230c739f434d7ea4d8b1d5f3721bc0f3558049b38f44"}, +] +pylint = [ + {file = "pylint-2.5.3-py3-none-any.whl", hash = "sha256:d0ece7d223fe422088b0e8f13fa0a1e8eb745ebffcb8ed53d3e95394b6101a1c"}, + {file = "pylint-2.5.3.tar.gz", hash = "sha256:7dd78437f2d8d019717dbf287772d0b2dbdfd13fc016aa7faa08d67bccc46adc"}, +] +pyyaml = [ + {file = "PyYAML-5.3.1-cp27-cp27m-win32.whl", hash = "sha256:74809a57b329d6cc0fdccee6318f44b9b8649961fa73144a98735b0aaf029f1f"}, + {file = "PyYAML-5.3.1-cp27-cp27m-win_amd64.whl", hash = "sha256:240097ff019d7c70a4922b6869d8a86407758333f02203e0fc6ff79c5dcede76"}, + {file = "PyYAML-5.3.1-cp35-cp35m-win32.whl", hash = "sha256:4f4b913ca1a7319b33cfb1369e91e50354d6f07a135f3b901aca02aa95940bd2"}, + {file = "PyYAML-5.3.1-cp35-cp35m-win_amd64.whl", hash = "sha256:cc8955cfbfc7a115fa81d85284ee61147059a753344bc51098f3ccd69b0d7e0c"}, + {file = "PyYAML-5.3.1-cp36-cp36m-win32.whl", hash = "sha256:7739fc0fa8205b3ee8808aea45e968bc90082c10aef6ea95e855e10abf4a37b2"}, + {file = "PyYAML-5.3.1-cp36-cp36m-win_amd64.whl", hash = "sha256:69f00dca373f240f842b2931fb2c7e14ddbacd1397d57157a9b005a6a9942648"}, + {file = "PyYAML-5.3.1-cp37-cp37m-win32.whl", hash = "sha256:d13155f591e6fcc1ec3b30685d50bf0711574e2c0dfffd7644babf8b5102ca1a"}, + {file = "PyYAML-5.3.1-cp37-cp37m-win_amd64.whl", hash = "sha256:73f099454b799e05e5ab51423c7bcf361c58d3206fa7b0d555426b1f4d9a3eaf"}, + {file = "PyYAML-5.3.1-cp38-cp38-win32.whl", hash = "sha256:06a0d7ba600ce0b2d2fe2e78453a470b5a6e000a985dd4a4e54e436cc36b0e97"}, + {file = "PyYAML-5.3.1-cp38-cp38-win_amd64.whl", hash = "sha256:95f71d2af0ff4227885f7a6605c37fd53d3a106fcab511b8860ecca9fcf400ee"}, + {file = "PyYAML-5.3.1.tar.gz", hash = "sha256:b8eac752c5e14d3eca0e6dd9199cd627518cb5ec06add0de9d32baeee6fe645d"}, +] regex = [ {file = "regex-2020.7.14-cp27-cp27m-win32.whl", hash = "sha256:e46d13f38cfcbb79bfdb2964b0fe12561fe633caf964a77a5f8d4e45fe5d2ef7"}, {file = "regex-2020.7.14-cp27-cp27m-win_amd64.whl", hash = "sha256:6961548bba529cac7c07af2fd4d527c5b91bb8fe18995fed6044ac22b3d14644"}, @@ -298,10 +987,29 @@ regex = [ {file = "regex-2020.7.14-cp38-cp38-win_amd64.whl", hash = "sha256:7a2dd66d2d4df34fa82c9dc85657c5e019b87932019947faece7983f2089a840"}, {file = "regex-2020.7.14.tar.gz", hash = "sha256:3a3af27a8d23143c49a3420efe5b3f8cf1a48c6fc8bc6856b03f638abc1833bb"}, ] +restructuredtext-lint = [ + {file = "restructuredtext_lint-1.3.1.tar.gz", hash = "sha256:470e53b64817211a42805c3a104d2216f6f5834b22fe7adb637d1de4d6501fb8"}, +] six = [ {file = "six-1.15.0-py2.py3-none-any.whl", hash = "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced"}, {file = "six-1.15.0.tar.gz", hash = "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259"}, ] +smmap = [ + {file = "smmap-3.0.4-py2.py3-none-any.whl", hash = "sha256:54c44c197c819d5ef1991799a7e30b662d1e520f2ac75c9efbeb54a742214cf4"}, + {file = "smmap-3.0.4.tar.gz", hash = "sha256:9c98bbd1f9786d22f14b3d4126894d56befb835ec90cef151af566c7e19b5d24"}, +] +snowballstemmer = [ + {file = "snowballstemmer-2.0.0-py2.py3-none-any.whl", hash = "sha256:209f257d7533fdb3cb73bdbd24f436239ca3b2fa67d56f6ff88e86be08cc5ef0"}, + {file = "snowballstemmer-2.0.0.tar.gz", hash = "sha256:df3bac3df4c2c01363f3dd2cfa78cce2840a79b9f1c2d2de9ce8d31683992f52"}, +] +stevedore = [ + {file = "stevedore-3.2.0-py3-none-any.whl", hash = "sha256:c8f4f0ebbc394e52ddf49de8bcc3cf8ad2b4425ebac494106bbc5e3661ac7633"}, + {file = "stevedore-3.2.0.tar.gz", hash = "sha256:38791aa5bed922b0a844513c5f9ed37774b68edc609e5ab8ab8d8fe0ce4315e5"}, +] +testfixtures = [ + {file = "testfixtures-6.14.1-py2.py3-none-any.whl", hash = "sha256:30566e24a1b34e4d3f8c13abf62557d01eeb4480bcb8f1745467bfb0d415a7d9"}, + {file = "testfixtures-6.14.1.tar.gz", hash = "sha256:58d2b3146d93bc5ddb0cd24e0ccacb13e29bdb61e5c81235c58f7b8ee4470366"}, +] toml = [ {file = "toml-0.10.1-py2.py3-none-any.whl", hash = "sha256:bda89d5935c2eac546d648028b9901107a595863cb36bae0c73ac804a9b4ce88"}, {file = "toml-0.10.1.tar.gz", hash = "sha256:926b612be1e5ce0634a2ca03470f95169cf16f939018233a670519cb4ac58b0f"}, @@ -329,7 +1037,19 @@ typed-ast = [ {file = "typed_ast-1.4.1-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:d43943ef777f9a1c42bf4e552ba23ac77a6351de620aa9acf64ad54933ad4d34"}, {file = "typed_ast-1.4.1.tar.gz", hash = "sha256:8c8aaad94455178e3187ab22c8b01a3837f8ee50e09cf31f1ba129eb293ec30b"}, ] +typing-extensions = [ + {file = "typing_extensions-3.7.4.2-py2-none-any.whl", hash = "sha256:f8d2bd89d25bc39dabe7d23df520442fa1d8969b82544370e03d88b5a591c392"}, + {file = "typing_extensions-3.7.4.2-py3-none-any.whl", hash = "sha256:6e95524d8a547a91e08f404ae485bbb71962de46967e1b71a0cb89af24e761c5"}, + {file = "typing_extensions-3.7.4.2.tar.gz", hash = "sha256:79ee589a3caca649a9bfd2a8de4709837400dfa00b6cc81962a1e6a1815969ae"}, +] virtualenv = [ {file = "virtualenv-20.0.29-py2.py3-none-any.whl", hash = "sha256:8aa9c37b082664dbce2236fa420759c02d64109d8e6013593ad13914718a30fd"}, {file = "virtualenv-20.0.29.tar.gz", hash = "sha256:f14a0a98ea4397f0d926cff950361766b6a73cd5975ae7eb259d12919f819a25"}, ] +wemake-python-styleguide = [ + {file = "wemake-python-styleguide-0.14.1.tar.gz", hash = "sha256:e13dc580fa56b7b548de8da170bccb8ddff2d4ab026ca987db8a9893bf8a7b5b"}, + {file = "wemake_python_styleguide-0.14.1-py3-none-any.whl", hash = "sha256:73a501e0547275287a2b926515c000cc25026a8bceb9dcc1bf73ef85a223a3c6"}, +] +wrapt = [ + {file = "wrapt-1.12.1.tar.gz", hash = "sha256:b62ffa81fb85f4332a4f609cab4ac40709470da05643a082ec1eb88e6d9b97d7"}, +] diff --git a/pyproject.toml b/pyproject.toml index 8fe4449..b63b74f 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -4,7 +4,7 @@ requires = ["poetry>=0.12"] [tool.black] line-length = 88 -skip-string-normalization = true +skip-string-normalization = true # wemake-python-styleguide enforces single quotes target-version = ["py38"] [tool.poetry] @@ -34,4 +34,13 @@ nox = "^2020.5.24" # Code Formatters autoflake = "^1.3.1" black = "^19.10b0" -isort = "^5.2.2" +isort = "^4.3.21" # TODO (isort): not ^5.2.2 due to pylint and wemake-python-styleguide + +# (Static) Code Analyzers +flake8 = "^3.8.3" +flake8-annotations = "^2.3.0" +flake8-black = "^0.2.1" +flake8-expression-complexity = "^0.0.8" +mypy = "^0.782" +pylint = "^2.5.3" +wemake-python-styleguide = "^0.14.1" # flake8 plug-in diff --git a/setup.cfg b/setup.cfg index 534a0f5..93815ad 100644 --- a/setup.cfg +++ b/setup.cfg @@ -2,6 +2,111 @@ # black's settings are in pyproject.toml => [tool.black] +[flake8] +# Include error classes only explicitly +# to avoid forward compatibility issues. +select = + # ============= + # flake8's base + # ============= + # mccabe => cyclomatic complexity + C901, + # pycodestyle => PEP8 compliance + E, W, + # pyflakes => basic errors + F4, F5, F6, F7, F8, F9 + # ======================== + # wemake-python-styleguide + # Source: https://wemake-python-stylegui.de/en/latest/pages/usage/violations/index.html + # ======================== + WPS1, WPS2, WPS3, WPS4, WPS5, WPS6, + # darglint => docstring matches implementation + DAR0, DAR1, DAR2, DAR3, DAR4, DAR5, + # flake8-bandit => common security issues + S1, S2, S3, S4, S5, S6, S7, + # flake8-broken-line => no \ to end a line + N400, + # flake8-bugbear => opinionated bugs and design flaws + B0, B3, B9, + # flake8-commas => better comma placements + C8, + # flake8-comprehensions => better comprehensions + C4, + # flake8-debugger => no debugger usage + T100, + # flake8-docstrings => PEP257 compliance + D1, D2, D3, D4, + # flake8-eradicate => no commented out code + E800, + # flake8-isort => isort would make changes + I0, + # flake8-rst-docstrings => valid rst in docstrings + RST2, RST3, RST4, + # flake8-string-format => unify usage of str.format() + P1, P2, P3, + # flake8-quotes => use double quotes everywhere (complying with black) + Q0, + # pep8-naming + N8, + # ===== + # other + # ===== + # flake8-annotations => enforce type checking for functions + ANN0, ANN2, ANN3, + # flake8-black => complain if black would make changes + BLK1, BLK9, + # flake8-expression-complexity => not too many expressions at once + ECE001, + +# By default, flake8 ignores some errors. +# Instead, do not ignore anything. +ignore = + +# If --ignore is passed on the command +# line, still ignore the following: +extend-ignore = + # Comply with black's style. + # Source: https://github.com/psf/black/blob/master/docs/compatible_configs.md#flake8 + E203, W503, + # f-strings are ok. + WPS305, + +per-file-ignores = + noxfile.py: + # TODO (isort): Check if still too many module members. + WPS202, + # TODO (isort): Remove after simplifying the nox session "lint". + WPS213, + # No overuse of string constants (e.g., '--version'). + WPS226, + +# Explicitly set mccabe's maximum complexity to 10 as recommended by +# Thomas McCabe, the inventor of the McCabe complexity, and the NIST. +# Source: https://en.wikipedia.org/wiki/Cyclomatic_complexity#Limiting_complexity_during_development +max-complexity = 10 + +# Comply with black's style. +# Source: https://github.com/psf/black/blob/master/docs/the_black_code_style.md#line-length +max-line-length = 88 + +# Preview the code lines that cause errors. +show-source = true + +# =================================== +# wemake-python-styleguide's settings +# =================================== +allowed-domain-names = + result, +min-name-length = 3 +max-name-length = 40 +# darglint +strictness = long +# flake8-docstrings +docstring-convention = google +# flake8-eradicate +eradicate-aggressive = true + + [isort] atomic = true case_sensitive = true @@ -24,3 +129,42 @@ use_parentheses = true # Source: https://google.github.io/styleguide/pyguide.html#313-imports-formatting force_single_line = true single_line_exclusions = typing + + +[mypy] +cache_dir = .cache/mypy + +[mypy-nox.*] +ignore_missing_imports = true + + +[pylint.FORMAT] +# Comply with black's style. +max-line-length = 88 + +[pylint.MESSAGES CONTROL] +disable = + # We use TODO's to indicate locations in the source base + # that must be worked on in the near future. + fixme, + # Comply with black's style. + bad-continuation, bad-whitespace, + # ===================== + # flake8 de-duplication + # Source: https://pylint.pycqa.org/en/latest/faq.html#i-am-using-another-popular-linter-alongside-pylint-which-messages-should-i-disable-to-avoid-duplicates + # ===================== + # mccabe + too-many-branches, + # pep8-naming + bad-classmethod-argument, bad-mcs-classmethod-argument, + invalid-name, no-self-argument, + # pycodestyle + bad-indentation, bare-except, line-too-long, missing-final-newline, + multiple-statements, trailing-whitespace, unnecessary-semicolon, unneeded-not, + # pydocstyle + missing-class-docstring, missing-function-docstring, missing-module-docstring, + # pyflakes + undefined-variable, unused-import, unused-variable, + +[pylint.REPORTS] +score = no From 9fc5b4816a743f133387949e7a5392b7215393dd Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Tue, 4 Aug 2020 00:09:29 +0200 Subject: [PATCH 04/13] Add a testing tool chain - use pytest as the base, measure coverage with pytest-cov + configure coverage to include branches and specify source locations + configure pytest to enforce explicit markers - add a package for the test suite under tests/ - add a `__version__` identifier at the package's root + it is dynamically assigned the version of the installed package + the version is PEP440 compliant and follows a strict subset of semantic versioning: x.y.z[.devN] where x, y, z, and N are all non-negative integers + add module with tests for the __version__ - add a nox session "test" that runs the test suite - use flake8 to lint pytest for consistent style --- noxfile.py | 26 +++- poetry.lock | 202 +++++++++++++++++++++++++++- pyproject.toml | 6 + setup.cfg | 50 ++++++- src/urban_meal_delivery/__init__.py | 8 ++ tests/__init__.py | 1 + tests/test_version.py | 116 ++++++++++++++++ 7 files changed, 406 insertions(+), 3 deletions(-) create mode 100644 tests/__init__.py create mode 100644 tests/test_version.py diff --git a/noxfile.py b/noxfile.py index 683ab79..9b24133 100644 --- a/noxfile.py +++ b/noxfile.py @@ -14,8 +14,11 @@ MAIN_PYTHON = '3.8' # Keep the project is forward compatible. NEXT_PYTHON = '3.9' +# Path to the test suite. +PYTEST_LOCATION = 'tests/' + # Paths with *.py files. -SRC_LOCATIONS = 'noxfile.py', 'src/' +SRC_LOCATIONS = 'noxfile.py', 'src/', PYTEST_LOCATION # Use a unified .cache/ folder for all tools. @@ -70,6 +73,7 @@ def lint(session: Session) -> None: 'flake8-annotations', 'flake8-black', 'flake8-expression-complexity', + 'flake8-pytest-style', 'mypy', 'pylint', 'wemake-python-styleguide', @@ -96,7 +100,27 @@ def lint(session: Session) -> None: @nox.session(python=[MAIN_PYTHON, NEXT_PYTHON]) def test(session: Session) -> None: """Test the code base.""" + # Re-using an old environment is not so easy here as + # `poetry install --no-dev` removes previously installed packages. + # We keep things simple and forbid such usage. + if session.virtualenv.reuse_existing: + raise RuntimeError('The "test" session must be run with the "-r" option') + _begin(session) + # Install only the non-develop dependencies + # and the testing tool chain. + session.run('poetry', 'install', '--no-dev', external=True) + _install_packages(session, 'pytest', 'pytest-cov') + # Interpret extra arguments as options for pytest. + args = session.posargs or ( + '--cov', + '--no-cov-on-fail', + '--cov-branch', + '--cov-fail-under=100', + '--cov-report=term-missing:skip-covered', + PYTEST_LOCATION, + ) + session.run('pytest', *args) @nox.session(name='pre-commit', python=MAIN_PYTHON, venv_backend='none') diff --git a/poetry.lock b/poetry.lock index 2c6ed11..f4a669c 100644 --- a/poetry.lock +++ b/poetry.lock @@ -49,6 +49,15 @@ lazy-object-proxy = ">=1.4.0,<1.5.0" six = ">=1.12,<2.0" wrapt = ">=1.11,<2.0" +[[package]] +category = "dev" +description = "Atomic file writes." +marker = "sys_platform == \"win32\"" +name = "atomicwrites" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "1.4.0" + [[package]] category = "dev" description = "Classes Without Boilerplate" @@ -137,6 +146,17 @@ version = "4.2.1" [package.dependencies] colorama = "*" +[[package]] +category = "dev" +description = "Code coverage measurement for Python" +name = "coverage" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, <4" +version = "5.2.1" + +[package.extras] +toml = ["toml"] + [[package]] category = "dev" description = "A utility for ensuring Google-style docstrings stay up to date with the source code." @@ -341,6 +361,14 @@ version = ">=4.3.5,<5" [package.extras] test = ["pytest (>=4.0.2,<6)"] +[[package]] +category = "dev" +description = "The package provides base classes and utils for flake8 plugin writing" +name = "flake8-plugin-utils" +optional = false +python-versions = ">=3.6,<4.0" +version = "1.3.0" + [[package]] category = "dev" description = "Polyfill package for Flake8 plugins" @@ -352,6 +380,17 @@ version = "1.0.2" [package.dependencies] flake8 = "*" +[[package]] +category = "dev" +description = "A flake8 plugin checking common style issues or inconsistencies with pytest-based tests." +name = "flake8-pytest-style" +optional = false +python-versions = ">=3.6,<4.0" +version = "1.2.2" + +[package.dependencies] +flake8-plugin-utils = ">=1.3.0,<2.0.0" + [[package]] category = "dev" description = "Flake8 lint for quotes." @@ -408,6 +447,14 @@ version = "3.1.7" [package.dependencies] gitdb = ">=4.0.1,<5" +[[package]] +category = "dev" +description = "iniconfig: brain-dead simple config-ini parsing" +name = "iniconfig" +optional = false +python-versions = "*" +version = "1.0.1" + [[package]] category = "dev" description = "A Python utility / library to sort Python imports." @@ -438,6 +485,14 @@ optional = false python-versions = "*" version = "0.6.1" +[[package]] +category = "dev" +description = "More routines for operating on iterables, beyond itertools" +name = "more-itertools" +optional = false +python-versions = ">=3.5" +version = "8.4.0" + [[package]] category = "dev" description = "Optional static typing for Python" @@ -479,6 +534,18 @@ virtualenv = ">=14.0.0" [package.extras] tox_to_nox = ["jinja2", "tox"] +[[package]] +category = "dev" +description = "Core utilities for Python packages" +name = "packaging" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "20.4" + +[package.dependencies] +pyparsing = ">=2.0.2" +six = "*" + [[package]] category = "dev" description = "Utility library for gitignore style pattern matching of file paths." @@ -506,6 +573,17 @@ version = "0.9.1" [package.dependencies] flake8-polyfill = ">=1.0.2,<2" +[[package]] +category = "dev" +description = "plugin and hook calling mechanisms for python" +name = "pluggy" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "0.13.1" + +[package.extras] +dev = ["pre-commit", "tox"] + [[package]] category = "dev" description = "library with cross-python path, ini-parsing, io, code, log facilities" @@ -564,6 +642,52 @@ isort = ">=4.2.5,<5" mccabe = ">=0.6,<0.7" toml = ">=0.7.1" +[[package]] +category = "dev" +description = "Python parsing module" +name = "pyparsing" +optional = false +python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" +version = "2.4.7" + +[[package]] +category = "dev" +description = "pytest: simple powerful testing with Python" +name = "pytest" +optional = false +python-versions = ">=3.5" +version = "6.0.1" + +[package.dependencies] +atomicwrites = ">=1.0" +attrs = ">=17.4.0" +colorama = "*" +iniconfig = "*" +more-itertools = ">=4.0.0" +packaging = "*" +pluggy = ">=0.12,<1.0" +py = ">=1.8.2" +toml = "*" + +[package.extras] +checkqa_mypy = ["mypy (0.780)"] +testing = ["argcomplete", "hypothesis (>=3.56)", "mock", "nose", "requests", "xmlschema"] + +[[package]] +category = "dev" +description = "Pytest plugin for measuring coverage." +name = "pytest-cov" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +version = "2.10.0" + +[package.dependencies] +coverage = ">=4.4" +pytest = ">=4.6" + +[package.extras] +testing = ["fields", "hunter", "process-tests (2.0.2)", "six", "pytest-xdist", "virtualenv"] + [[package]] category = "dev" description = "YAML parser and emitter for Python" @@ -719,7 +843,7 @@ python-versions = "*" version = "1.12.1" [metadata] -content-hash = "e69534c9133ef3fb975f33937c97711f69281a8ffb931097a7b9e4ae1132fa09" +content-hash = "1a6ddbc4c05cb41329cbac4792210897f11dda8e8ba3aa7e814a4a0d0d9c4fa8" lock-version = "1.0" python-versions = "^3.8" @@ -744,6 +868,10 @@ astroid = [ {file = "astroid-2.4.2-py3-none-any.whl", hash = "sha256:bc58d83eb610252fd8de6363e39d4f1d0619c894b0ed24603b881c02e64c7386"}, {file = "astroid-2.4.2.tar.gz", hash = "sha256:2f4078c2a41bf377eea06d71c9d2ba4eb8f6b1af2135bec27bbbb7d8f12bb703"}, ] +atomicwrites = [ + {file = "atomicwrites-1.4.0-py2.py3-none-any.whl", hash = "sha256:6d1784dea7c0c8d4a5172b6c620f40b6e4cbfdf96d783691f2e1302a7b88e197"}, + {file = "atomicwrites-1.4.0.tar.gz", hash = "sha256:ae70396ad1a434f9c7046fd2dd196fc04b12f9e91ffb859164193be8b6168a7a"}, +] attrs = [ {file = "attrs-19.3.0-py2.py3-none-any.whl", hash = "sha256:08a96c641c3a74e44eb59afb61a24f2cb9f4d7188748e76ba4bb5edfa3cb7d1c"}, {file = "attrs-19.3.0.tar.gz", hash = "sha256:f7b7ce16570fe9965acd6d30101a28f62fb4a7f9e926b3bbc9b61f8b04247e72"}, @@ -771,6 +899,42 @@ colorlog = [ {file = "colorlog-4.2.1-py2.py3-none-any.whl", hash = "sha256:43597fd822ce705190fc997519342fdaaf44b9b47f896ece7aa153ed4b909c74"}, {file = "colorlog-4.2.1.tar.gz", hash = "sha256:75e55822c3a3387d721579241e776de2cf089c9ef9528b1f09e8b04d403ad118"}, ] +coverage = [ + {file = "coverage-5.2.1-cp27-cp27m-macosx_10_13_intel.whl", hash = "sha256:40f70f81be4d34f8d491e55936904db5c527b0711b2a46513641a5729783c2e4"}, + {file = "coverage-5.2.1-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:675192fca634f0df69af3493a48224f211f8db4e84452b08d5fcebb9167adb01"}, + {file = "coverage-5.2.1-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:2fcc8b58953d74d199a1a4d633df8146f0ac36c4e720b4a1997e9b6327af43a8"}, + {file = "coverage-5.2.1-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:64c4f340338c68c463f1b56e3f2f0423f7b17ba6c3febae80b81f0e093077f59"}, + {file = "coverage-5.2.1-cp27-cp27m-win32.whl", hash = "sha256:52f185ffd3291196dc1aae506b42e178a592b0b60a8610b108e6ad892cfc1bb3"}, + {file = "coverage-5.2.1-cp27-cp27m-win_amd64.whl", hash = "sha256:30bc103587e0d3df9e52cd9da1dd915265a22fad0b72afe54daf840c984b564f"}, + {file = "coverage-5.2.1-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:9ea749fd447ce7fb1ac71f7616371f04054d969d412d37611716721931e36efd"}, + {file = "coverage-5.2.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:ce7866f29d3025b5b34c2e944e66ebef0d92e4a4f2463f7266daa03a1332a651"}, + {file = "coverage-5.2.1-cp35-cp35m-macosx_10_13_x86_64.whl", hash = "sha256:4869ab1c1ed33953bb2433ce7b894a28d724b7aa76c19b11e2878034a4e4680b"}, + {file = "coverage-5.2.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:a3ee9c793ffefe2944d3a2bd928a0e436cd0ac2d9e3723152d6fd5398838ce7d"}, + {file = "coverage-5.2.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:28f42dc5172ebdc32622a2c3f7ead1b836cdbf253569ae5673f499e35db0bac3"}, + {file = "coverage-5.2.1-cp35-cp35m-win32.whl", hash = "sha256:e26c993bd4b220429d4ec8c1468eca445a4064a61c74ca08da7429af9bc53bb0"}, + {file = "coverage-5.2.1-cp35-cp35m-win_amd64.whl", hash = "sha256:4186fc95c9febeab5681bc3248553d5ec8c2999b8424d4fc3a39c9cba5796962"}, + {file = "coverage-5.2.1-cp36-cp36m-macosx_10_13_x86_64.whl", hash = "sha256:b360d8fd88d2bad01cb953d81fd2edd4be539df7bfec41e8753fe9f4456a5082"}, + {file = "coverage-5.2.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:1adb6be0dcef0cf9434619d3b892772fdb48e793300f9d762e480e043bd8e716"}, + {file = "coverage-5.2.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:098a703d913be6fbd146a8c50cc76513d726b022d170e5e98dc56d958fd592fb"}, + {file = "coverage-5.2.1-cp36-cp36m-win32.whl", hash = "sha256:962c44070c281d86398aeb8f64e1bf37816a4dfc6f4c0f114756b14fc575621d"}, + {file = "coverage-5.2.1-cp36-cp36m-win_amd64.whl", hash = "sha256:b1ed2bdb27b4c9fc87058a1cb751c4df8752002143ed393899edb82b131e0546"}, + {file = "coverage-5.2.1-cp37-cp37m-macosx_10_13_x86_64.whl", hash = "sha256:c890728a93fffd0407d7d37c1e6083ff3f9f211c83b4316fae3778417eab9811"}, + {file = "coverage-5.2.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:538f2fd5eb64366f37c97fdb3077d665fa946d2b6d95447622292f38407f9258"}, + {file = "coverage-5.2.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:27ca5a2bc04d68f0776f2cdcb8bbd508bbe430a7bf9c02315cd05fb1d86d0034"}, + {file = "coverage-5.2.1-cp37-cp37m-win32.whl", hash = "sha256:aab75d99f3f2874733946a7648ce87a50019eb90baef931698f96b76b6769a46"}, + {file = "coverage-5.2.1-cp37-cp37m-win_amd64.whl", hash = "sha256:c2ff24df02a125b7b346c4c9078c8936da06964cc2d276292c357d64378158f8"}, + {file = "coverage-5.2.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:304fbe451698373dc6653772c72c5d5e883a4aadaf20343592a7abb2e643dae0"}, + {file = "coverage-5.2.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:c96472b8ca5dc135fb0aa62f79b033f02aa434fb03a8b190600a5ae4102df1fd"}, + {file = "coverage-5.2.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:8505e614c983834239f865da2dd336dcf9d72776b951d5dfa5ac36b987726e1b"}, + {file = "coverage-5.2.1-cp38-cp38-win32.whl", hash = "sha256:700997b77cfab016533b3e7dbc03b71d33ee4df1d79f2463a318ca0263fc29dd"}, + {file = "coverage-5.2.1-cp38-cp38-win_amd64.whl", hash = "sha256:46794c815e56f1431c66d81943fa90721bb858375fb36e5903697d5eef88627d"}, + {file = "coverage-5.2.1-cp39-cp39-macosx_10_13_x86_64.whl", hash = "sha256:16042dc7f8e632e0dcd5206a5095ebd18cb1d005f4c89694f7f8aafd96dd43a3"}, + {file = "coverage-5.2.1-cp39-cp39-manylinux1_i686.whl", hash = "sha256:c1bbb628ed5192124889b51204de27c575b3ffc05a5a91307e7640eff1d48da4"}, + {file = "coverage-5.2.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:4f6428b55d2916a69f8d6453e48a505c07b2245653b0aa9f0dee38785939f5e4"}, + {file = "coverage-5.2.1-cp39-cp39-win32.whl", hash = "sha256:9e536783a5acee79a9b308be97d3952b662748c4037b6a24cbb339dc7ed8eb89"}, + {file = "coverage-5.2.1-cp39-cp39-win_amd64.whl", hash = "sha256:b8f58c7db64d8f27078cbf2a4391af6aa4e4767cc08b37555c4ae064b8558d9b"}, + {file = "coverage-5.2.1.tar.gz", hash = "sha256:a34cb28e0747ea15e82d13e14de606747e9e484fb28d63c999483f5d5188e89b"}, +] darglint = [ {file = "darglint-1.5.2-py3-none-any.whl", hash = "sha256:049a98cf3aec8cf6ea344a863c68112d80b7f8de214459b5fa6853371f89c3e7"}, {file = "darglint-1.5.2.tar.gz", hash = "sha256:6b9461f96694c2cf1d8edb1597a783fe6840953b0eb18cc6cc1e72a26f196d79"}, @@ -838,10 +1002,18 @@ flake8-isort = [ {file = "flake8-isort-3.0.1.tar.gz", hash = "sha256:5d976da513cc390232ad5a9bb54aee8a092466a15f442d91dfc525834bee727a"}, {file = "flake8_isort-3.0.1-py2.py3-none-any.whl", hash = "sha256:df1dd6dd73f6a8b128c9c783356627231783cccc82c13c6dc343d1a5a491699b"}, ] +flake8-plugin-utils = [ + {file = "flake8-plugin-utils-1.3.0.tar.gz", hash = "sha256:965931e7c17a760915e38bb10dc60516b414ef8210e987252a8d73dcb196a5f5"}, + {file = "flake8_plugin_utils-1.3.0-py3-none-any.whl", hash = "sha256:305461c4fbf94877bcc9ccf435771b135d72a40eefd92e70a4b5f761ca43b1c8"}, +] flake8-polyfill = [ {file = "flake8-polyfill-1.0.2.tar.gz", hash = "sha256:e44b087597f6da52ec6393a709e7108b2905317d0c0b744cdca6208e670d8eda"}, {file = "flake8_polyfill-1.0.2-py2.py3-none-any.whl", hash = "sha256:12be6a34ee3ab795b19ca73505e7b55826d5f6ad7230d31b18e106400169b9e9"}, ] +flake8-pytest-style = [ + {file = "flake8-pytest-style-1.2.2.tar.gz", hash = "sha256:93830b18d961613a012061867b15b1434e433662197ec3a7594db8389637d5f9"}, + {file = "flake8_pytest_style-1.2.2-py3-none-any.whl", hash = "sha256:63e93d10b9cca6c73309319e89e7b1a8d316c7d1f12407faca975c498192fde3"}, +] flake8-quotes = [ {file = "flake8-quotes-2.1.2.tar.gz", hash = "sha256:c844c9592940c8926c60f00bc620808912ff2acd34923ab5338f3a5ca618a331"}, ] @@ -860,6 +1032,10 @@ gitpython = [ {file = "GitPython-3.1.7-py3-none-any.whl", hash = "sha256:fa3b92da728a457dd75d62bb5f3eb2816d99a7fe6c67398e260637a40e3fafb5"}, {file = "GitPython-3.1.7.tar.gz", hash = "sha256:2db287d71a284e22e5c2846042d0602465c7434d910406990d5b74df4afb0858"}, ] +iniconfig = [ + {file = "iniconfig-1.0.1-py3-none-any.whl", hash = "sha256:80cf40c597eb564e86346103f609d74efce0f6b4d4f30ec8ce9e2c26411ba437"}, + {file = "iniconfig-1.0.1.tar.gz", hash = "sha256:e5f92f89355a67de0595932a6c6c02ab4afddc6fcdc0bfc5becd0d60884d3f69"}, +] isort = [ {file = "isort-4.3.21-py2.py3-none-any.whl", hash = "sha256:6e811fcb295968434526407adb8796944f1988c5b65e8139058f2014cbe100fd"}, {file = "isort-4.3.21.tar.gz", hash = "sha256:54da7e92468955c4fceacd0c86bd0ec997b0e1ee80d97f67c35a78b719dccab1"}, @@ -891,6 +1067,10 @@ mccabe = [ {file = "mccabe-0.6.1-py2.py3-none-any.whl", hash = "sha256:ab8a6258860da4b6677da4bd2fe5dc2c659cff31b3ee4f7f5d64e79735b80d42"}, {file = "mccabe-0.6.1.tar.gz", hash = "sha256:dd8d182285a0fe56bace7f45b5e7d1a6ebcbf524e8f3bd87eb0f125271b8831f"}, ] +more-itertools = [ + {file = "more-itertools-8.4.0.tar.gz", hash = "sha256:68c70cc7167bdf5c7c9d8f6954a7837089c6a36bf565383919bb595efb8a17e5"}, + {file = "more_itertools-8.4.0-py3-none-any.whl", hash = "sha256:b78134b2063dd214000685165d81c154522c3ee0a1c0d4d113c80361c234c5a2"}, +] mypy = [ {file = "mypy-0.782-cp35-cp35m-macosx_10_6_x86_64.whl", hash = "sha256:2c6cde8aa3426c1682d35190b59b71f661237d74b053822ea3d748e2c9578a7c"}, {file = "mypy-0.782-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:9c7a9a7ceb2871ba4bac1cf7217a7dd9ccd44c27c2950edbc6dc08530f32ad4e"}, @@ -915,6 +1095,10 @@ nox = [ {file = "nox-2020.5.24-py3-none-any.whl", hash = "sha256:c4509621fead99473a1401870e680b0aadadce5c88440f0532863595176d64c1"}, {file = "nox-2020.5.24.tar.gz", hash = "sha256:61a55705736a1a73efbd18d5b262a43d55a1176546e0eb28b29064cfcffe26c0"}, ] +packaging = [ + {file = "packaging-20.4-py2.py3-none-any.whl", hash = "sha256:998416ba6962ae7fbd6596850b80e17859a5753ba17c32284f67bfff33784181"}, + {file = "packaging-20.4.tar.gz", hash = "sha256:4357f74f47b9c12db93624a82154e9b120fa8293699949152b22065d556079f8"}, +] pathspec = [ {file = "pathspec-0.8.0-py2.py3-none-any.whl", hash = "sha256:7d91249d21749788d07a2d0f94147accd8f845507400749ea19c1ec9054a12b0"}, {file = "pathspec-0.8.0.tar.gz", hash = "sha256:da45173eb3a6f2a5a487efba21f050af2b41948be6ab52b6a1e3ff22bb8b7061"}, @@ -927,6 +1111,10 @@ pep8-naming = [ {file = "pep8-naming-0.9.1.tar.gz", hash = "sha256:a33d38177056321a167decd6ba70b890856ba5025f0a8eca6a3eda607da93caf"}, {file = "pep8_naming-0.9.1-py2.py3-none-any.whl", hash = "sha256:45f330db8fcfb0fba57458c77385e288e7a3be1d01e8ea4268263ef677ceea5f"}, ] +pluggy = [ + {file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"}, + {file = "pluggy-0.13.1.tar.gz", hash = "sha256:15b2acde666561e1298d71b523007ed7364de07029219b604cf808bfa1c765b0"}, +] py = [ {file = "py-1.9.0-py2.py3-none-any.whl", hash = "sha256:366389d1db726cd2fcfc79732e75410e5fe4d31db13692115529d34069a043c2"}, {file = "py-1.9.0.tar.gz", hash = "sha256:9ca6883ce56b4e8da7e79ac18787889fa5206c79dcc67fb065376cd2fe03f342"}, @@ -951,6 +1139,18 @@ pylint = [ {file = "pylint-2.5.3-py3-none-any.whl", hash = "sha256:d0ece7d223fe422088b0e8f13fa0a1e8eb745ebffcb8ed53d3e95394b6101a1c"}, {file = "pylint-2.5.3.tar.gz", hash = "sha256:7dd78437f2d8d019717dbf287772d0b2dbdfd13fc016aa7faa08d67bccc46adc"}, ] +pyparsing = [ + {file = "pyparsing-2.4.7-py2.py3-none-any.whl", hash = "sha256:ef9d7589ef3c200abe66653d3f1ab1033c3c419ae9b9bdb1240a85b024efc88b"}, + {file = "pyparsing-2.4.7.tar.gz", hash = "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1"}, +] +pytest = [ + {file = "pytest-6.0.1-py3-none-any.whl", hash = "sha256:8b6007800c53fdacd5a5c192203f4e531eb2a1540ad9c752e052ec0f7143dbad"}, + {file = "pytest-6.0.1.tar.gz", hash = "sha256:85228d75db9f45e06e57ef9bf4429267f81ac7c0d742cc9ed63d09886a9fe6f4"}, +] +pytest-cov = [ + {file = "pytest-cov-2.10.0.tar.gz", hash = "sha256:1a629dc9f48e53512fcbfda6b07de490c374b0c83c55ff7a1720b3fccff0ac87"}, + {file = "pytest_cov-2.10.0-py2.py3-none-any.whl", hash = "sha256:6e6d18092dce6fad667cd7020deed816f858ad3b49d5b5e2b1cc1c97a4dba65c"}, +] pyyaml = [ {file = "PyYAML-5.3.1-cp27-cp27m-win32.whl", hash = "sha256:74809a57b329d6cc0fdccee6318f44b9b8649961fa73144a98735b0aaf029f1f"}, {file = "PyYAML-5.3.1-cp27-cp27m-win_amd64.whl", hash = "sha256:240097ff019d7c70a4922b6869d8a86407758333f02203e0fc6ff79c5dcede76"}, diff --git a/pyproject.toml b/pyproject.toml index b63b74f..c4934e0 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -41,6 +41,12 @@ flake8 = "^3.8.3" flake8-annotations = "^2.3.0" flake8-black = "^0.2.1" flake8-expression-complexity = "^0.0.8" +flake8-pytest-style = "^1.2.2" mypy = "^0.782" pylint = "^2.5.3" wemake-python-styleguide = "^0.14.1" # flake8 plug-in + +# Test Suite +packaging = "^20.4" # used to test the packaged version +pytest = "^6.0.1" +pytest-cov = "^2.10.0" diff --git a/setup.cfg b/setup.cfg index 93815ad..59af783 100644 --- a/setup.cfg +++ b/setup.cfg @@ -2,6 +2,23 @@ # black's settings are in pyproject.toml => [tool.black] +[coverage:paths] +source = + src/ + */site-packages/ + +[coverage:report] +show_missing = true +skip_covered = true +skip_empty = true + +[coverage:run] +branch = true +data_file = .cache/coverage/data +source = + urban_meal_delivery + + [flake8] # Include error classes only explicitly # to avoid forward compatibility issues. @@ -57,6 +74,8 @@ select = BLK1, BLK9, # flake8-expression-complexity => not too many expressions at once ECE001, + # flake8-pytest-style => enforce a consistent style with pytest + PT0, # By default, flake8 ignores some errors. # Instead, do not ignore anything. @@ -70,6 +89,10 @@ extend-ignore = E203, W503, # f-strings are ok. WPS305, + # Classes should not have to specify a base class. + WPS306, + # Putting logic into __init__.py files may be justified. + WPS412, per-file-ignores = noxfile.py: @@ -79,6 +102,11 @@ per-file-ignores = WPS213, # No overuse of string constants (e.g., '--version'). WPS226, + tests/*.py: + # `assert` statements are ok in the test suite. + S101, + # Shadowing outer scopes occurs naturally with mocks. + WPS442, # Explicitly set mccabe's maximum complexity to 10 as recommended by # Thomas McCabe, the inventor of the McCabe complexity, and the NIST. @@ -106,6 +134,17 @@ docstring-convention = google # flake8-eradicate eradicate-aggressive = true +# ============================== +# flake8-pytest-style's settings +# ============================== +# Prefer @pytest.fixture over @pytest.fixture(). +pytest-fixture-no-parentheses = true +# Prefer @pytest.mark.parametrize(['param1', 'param2'], [(1, 2), (3, 4)]) +# over @pytest.mark.parametrize(('param1', 'param2'), ([1, 2], [3, 4])) +pytest-parametrize-names-type = list +pytest-parametrize-values-row-type = tuple +pytest-parametrize-values-type = list + [isort] atomic = true @@ -134,7 +173,7 @@ single_line_exclusions = typing [mypy] cache_dir = .cache/mypy -[mypy-nox.*] +[mypy-nox.*,packaging.*,pytest] ignore_missing_imports = true @@ -165,6 +204,15 @@ disable = missing-class-docstring, missing-function-docstring, missing-module-docstring, # pyflakes undefined-variable, unused-import, unused-variable, + # wemake-python-styleguide + redefined-outer-name, [pylint.REPORTS] score = no + + +[tool:pytest] +addopts = + --strict-markers +cache_dir = .cache/pytest +console_output_style = count diff --git a/src/urban_meal_delivery/__init__.py b/src/urban_meal_delivery/__init__.py index 27023c3..d17278b 100644 --- a/src/urban_meal_delivery/__init__.py +++ b/src/urban_meal_delivery/__init__.py @@ -1 +1,9 @@ """Source code for the urban-meal-delivery research project.""" + +from importlib import metadata + + +try: + __version__ = metadata.version(__name__) +except metadata.PackageNotFoundError: # pragma: no cover + __version__ = 'unknown' diff --git a/tests/__init__.py b/tests/__init__.py new file mode 100644 index 0000000..606a650 --- /dev/null +++ b/tests/__init__.py @@ -0,0 +1 @@ +"""Test the urban-meal-delivery package.""" diff --git a/tests/test_version.py b/tests/test_version.py new file mode 100644 index 0000000..e3bf23c --- /dev/null +++ b/tests/test_version.py @@ -0,0 +1,116 @@ +"""Test the package's version identifier. + +This packaged version identifier must adhere to PEP440 +and a strict subset of semantic versioning: + +- version identifiers must follow the x.y.z format + where x.y.z are non-negative integers +- non-final versions are only allowed with a .devN suffix + where N is also a non-negative integer +""" + +import re + +import pytest +from packaging import version as pkg_version +from packaging.version import Version + +import urban_meal_delivery + + +@pytest.fixture +def parsed_version() -> str: + """The packaged version.""" + return pkg_version.Version(urban_meal_delivery.__version__) # noqa:WPS609 + + +class TestPEP404Compliance: + """Packaged version identifier is PEP440 compliant.""" + + # pylint:disable=no-self-use + + def test_parsed_version_has_no_epoch(self, parsed_version: Version) -> None: + """PEP440 compliant subset of semantic versioning: no epoch.""" + assert parsed_version.epoch == 0 + + def test_parsed_version_is_non_local(self, parsed_version: Version) -> None: + """PEP440 compliant subset of semantic versioning: no local version.""" + assert parsed_version.local is None + + def test_parsed_version_is_no_post_release(self, parsed_version: Version) -> None: + """PEP440 compliant subset of semantic versioning: no post releases.""" + assert parsed_version.is_postrelease is False + + def test_parsed_version_is_all_public(self, parsed_version: Version) -> None: + """PEP440 compliant subset of semantic versioning: all public parts.""" + assert parsed_version.public == urban_meal_delivery.__version__ # noqa:WPS609 + + +class TestSemanticVersioning: + """Packaged version follows a strict subset of semantic versioning.""" + + # pylint:disable=no-self-use + + version_pattern = re.compile( + r'^(0|([1-9]\d*))\.(0|([1-9]\d*))\.(0|([1-9]\d*))(\.dev(0|([1-9]\d*)))?$', + ) + + def test_version_is_semantic(self) -> None: + """Packaged version follows semantic versioning.""" + result = self.version_pattern.fullmatch( + urban_meal_delivery.__version__, # noqa:WPS609 + ) + + assert result is not None + + # The next two test cases are sanity checks to validate the version_pattern. + + @pytest.mark.parametrize( + 'version', + [ + '0.1.0', + '1.0.0', + '1.2.3', + '1.23.456', + '123.4.56', + '10.11.12', + '1.2.3.dev0', + '1.2.3.dev1', + '1.2.3.dev10', + ], + ) + def test_valid_semantic_versioning(self, version: str) -> None: + """Versions follow the x.y.z or x.y.z.devN format.""" + result = self.version_pattern.fullmatch(version) + + assert result is not None + + @pytest.mark.parametrize( + 'version', + [ + '1', + '1.2', + '-1.2.3', + '01.2.3', + '1.02.3', + '1.2.03', + '1.2.3.4', + '1.2.3.abc', + '1.2.3-dev0', + '1.2.3+dev0', + '1.2.3.d0', + '1.2.3.develop0', + '1.2.3.dev-1', + '1.2.3.dev01', + '1.2.3..dev0', + '1-2-3', + '1,2,3', + '1..2.3', + '1.2..3', + ], + ) + def test_invalid_semantic_versioning(self, version: str) -> None: + """Versions follow the x.y.z or x.y.z.devN format.""" + result = self.version_pattern.fullmatch(version) + + assert result is None From da233e2e35445080c502ba47b5c34573e255b927 Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Tue, 4 Aug 2020 02:55:41 +0200 Subject: [PATCH 05/13] Set up pre-commit hooks - add pre-commit and pre-merge hooks: + run `poetry run nox -s pre-commit` on staged *.py files + run common pre-commit hooks for validations that could not be achieved with tools in the develop environment so easily + run `poetry run nox -s pre-merge` before merges and pushes - implement the "pre-commit" and "pre-merge" sessions in nox + include a little hack to deal with the positional arguments passed by the pre-commit framework - provide more documentation on the nox sessions --- .pre-commit-config.yaml | 38 ++++++++++++++++++ noxfile.py | 87 ++++++++++++++++++++++++++++++++++------- poetry.lock | 60 +++++++++++++++++++++++++++- pyproject.toml | 1 + 4 files changed, 171 insertions(+), 15 deletions(-) create mode 100644 .pre-commit-config.yaml diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml new file mode 100644 index 0000000..cefbf7e --- /dev/null +++ b/.pre-commit-config.yaml @@ -0,0 +1,38 @@ +default_stages: [commit] # only used if a hook does not specify stages +fail_fast: true +repos: +# Run the local formatting, linting, and testing tool chains. +- repo: local + hooks: + - id: local-pre-commit-checks + name: Run code formatters and linters + entry: poetry run nox -s pre-commit -- + language: system + stages: [commit] + types: [python] + - id: local-pre-merge-checks + name: Run the entire test suite + entry: poetry run nox -s pre-merge -- + language: system + stages: [merge-commit, push] + types: [python] +# Enable hooks provided by the pre-commit project to +# enforce rules that local tools could not that easily. +- repo: https://github.com/pre-commit/pre-commit-hooks + rev: v3.2.0 + hooks: + - id: check-added-large-files + args: [--maxkb=100] + - id: check-case-conflict + - id: check-builtin-literals + - id: check-merge-conflict + - id: check-toml + - id: check-yaml + - id: end-of-file-fixer + stages: [commit] # overwrite the default + - id: mixed-line-ending + args: [--fix=no] + - id: no-commit-to-branch + args: [--branch, main] + - id: trailing-whitespace + stages: [commit] # overwrite the default diff --git a/noxfile.py b/noxfile.py index 9b24133..a1a09d6 100644 --- a/noxfile.py +++ b/noxfile.py @@ -39,7 +39,11 @@ nox.options.sessions = ( @nox.session(name='format', python=MAIN_PYTHON) def format_(session: Session) -> None: - """Format source files with autoflake, black, and isort.""" + """Format source files with autoflake, black, and isort. + + If no extra arguments are provided, all source files are formatted. + Otherwise, they are interpreted as paths the formatters work on recursively. + """ _begin(session) _install_packages(session, 'autoflake', 'black', 'isort') # Interpret extra arguments as locations of source files. @@ -65,7 +69,11 @@ def format_(session: Session) -> None: @nox.session(python=MAIN_PYTHON) def lint(session: Session) -> None: - """Lint source files with flake8, mypy, and pylint.""" + """Lint source files with flake8, mypy, and pylint. + + If no extra arguments are provided, all source files are linted. + Otherwise, they are interpreted as paths the linters work on recursively. + """ _begin(session) _install_packages( session, @@ -99,20 +107,33 @@ def lint(session: Session) -> None: @nox.session(python=[MAIN_PYTHON, NEXT_PYTHON]) def test(session: Session) -> None: - """Test the code base.""" + """Test the code base. + + Runs the unit and integration tests (written with pytest). + + If no extra arguments are provided, the entire test suite + is exexcuted and succeeds only with 100% coverage. + + If extra arguments are provided, they are + forwarded to pytest without any changes. + """ # Re-using an old environment is not so easy here as # `poetry install --no-dev` removes previously installed packages. # We keep things simple and forbid such usage. if session.virtualenv.reuse_existing: - raise RuntimeError('The "test" session must be run with the "-r" option') + raise RuntimeError( + 'The "test" and "pre-merge" sessions must be run without the "-r" option', + ) _begin(session) - # Install only the non-develop dependencies - # and the testing tool chain. + # Install only the non-develop dependencies and the testing tool chain. session.run('poetry', 'install', '--no-dev', external=True) _install_packages(session, 'pytest', 'pytest-cov') # Interpret extra arguments as options for pytest. - args = session.posargs or ( + # They are "dropped" by the hack in the pre_merge() function + # if this function is run within the "pre-merge" session. + posargs = () if session.env.get('_drop_posargs') else session.posargs + args = posargs or ( '--cov', '--no-cov-on-fail', '--cov-branch', @@ -125,23 +146,54 @@ def test(session: Session) -> None: @nox.session(name='pre-commit', python=MAIN_PYTHON, venv_backend='none') def pre_commit(session: Session) -> None: - """Source files must be well-formed before they enter git.""" - _begin(session) + """Source files must be well-formed before they enter git. + + Intended to be run as a pre-commit hook. + + This session is a wrapper that triggers the "format" and "lint" sessions. + + Passed in extra arguments are forwarded. So, if it is run as a pre-commit + hook, only the currently staged source files are formatted and linted. + """ + # "format" and "lint" are run in sessions on their own as + # session.notify() creates new Session objects. session.notify('format') session.notify('lint') -@nox.session(name='pre-merge', python=MAIN_PYTHON, venv_backend='none') +@nox.session(name='pre-merge', python=MAIN_PYTHON) def pre_merge(session: Session) -> None: - """The test suite must pass before merges are made.""" - _begin(session) - session.notify('test') + """The test suite must pass before merges are made. + + Intended to be run either as a pre-merge or pre-push hook. + + First, this session triggers the "format" and "lint" sessions via + the "pre-commit" session. + + Then, it runs the "test" session ignoring any extra arguments passed in + so that the entire test suite is executed. + """ + session.notify('pre-commit') + # Little hack to not work with the extra arguments provided + # by the pre-commit framework. Create a flag in the + # env(ironment) that must contain only `str`-like objects. + session.env['_drop_posargs'] = 'true' + # Cannot use session.notify() to trigger the "test" session + # as that would create a new Session object without the flag + # in the env(ironment). Instead, run the test() function within + # the "pre-merge" session. + test(session) def _begin(session: Session) -> None: """Show generic info about a session.""" if session.posargs: - print('extra arguments:', *session.posargs) # noqa:WPS421 + # Part of the hack in pre_merge() to "drop" the extra arguments. + # Indicate to stdout that the passed in extra arguments are ignored. + if session.env.get('_drop_posargs') is None: + print('Provided extra argument(s):', *session.posargs) # noqa:WPS421 + else: + print('The provided extra arguments are ignored') # noqa:WPS421 session.run('python', '--version') @@ -159,6 +211,13 @@ def _install_packages( packages respecting the pinnned versions specified in poetry's lock file. This makes nox sessions even more deterministic. + IMPORTANT: This function skips installation if the current nox session + is run with the "-r" flag to re-use an existing virtual environment. + That turns nox into a fast task runner provided that a virtual + environment actually existed and does not need to be changed (e.g., + new dependencies added in the meantime). Do not use the "-r" flag on CI + or as part of pre-commit hooks! + Args: session: the Session object *packages_or_pip_args: the packages to be installed or pip options diff --git a/poetry.lock b/poetry.lock index f4a669c..6e55d75 100644 --- a/poetry.lock +++ b/poetry.lock @@ -118,6 +118,14 @@ typed-ast = ">=1.4.0" [package.extras] d = ["aiohttp (>=3.3.2)", "aiohttp-cors"] +[[package]] +category = "dev" +description = "Validate configuration and produce human readable error messages." +name = "cfgv" +optional = false +python-versions = ">=3.6.1" +version = "3.2.0" + [[package]] category = "dev" description = "Composable command line interface toolkit" @@ -447,6 +455,17 @@ version = "3.1.7" [package.dependencies] gitdb = ">=4.0.1,<5" +[[package]] +category = "dev" +description = "File identification library for Python" +name = "identify" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,>=2.7" +version = "1.4.25" + +[package.extras] +license = ["editdistance"] + [[package]] category = "dev" description = "iniconfig: brain-dead simple config-ini parsing" @@ -517,6 +536,14 @@ optional = false python-versions = "*" version = "0.4.3" +[[package]] +category = "dev" +description = "Node.js virtual environment builder" +name = "nodeenv" +optional = false +python-versions = "*" +version = "1.4.0" + [[package]] category = "dev" description = "Flexible test automation." @@ -584,6 +611,22 @@ version = "0.13.1" [package.extras] dev = ["pre-commit", "tox"] +[[package]] +category = "dev" +description = "A framework for managing and maintaining multi-language pre-commit hooks." +name = "pre-commit" +optional = false +python-versions = ">=3.6.1" +version = "2.6.0" + +[package.dependencies] +cfgv = ">=2.0.0" +identify = ">=1.0.0" +nodeenv = ">=0.11.1" +pyyaml = ">=5.1" +toml = "*" +virtualenv = ">=20.0.8" + [[package]] category = "dev" description = "library with cross-python path, ini-parsing, io, code, log facilities" @@ -843,7 +886,7 @@ python-versions = "*" version = "1.12.1" [metadata] -content-hash = "1a6ddbc4c05cb41329cbac4792210897f11dda8e8ba3aa7e814a4a0d0d9c4fa8" +content-hash = "7a843263817a908ca01d198402d3e9310c33307ac85085f028c8fbdd7587f48f" lock-version = "1.0" python-versions = "^3.8" @@ -887,6 +930,10 @@ black = [ {file = "black-19.10b0-py36-none-any.whl", hash = "sha256:1b30e59be925fafc1ee4565e5e08abef6b03fe455102883820fe5ee2e4734e0b"}, {file = "black-19.10b0.tar.gz", hash = "sha256:c2edb73a08e9e0e6f65a0e6af18b059b8b1cdd5bef997d7a0b181df93dc81539"}, ] +cfgv = [ + {file = "cfgv-3.2.0-py2.py3-none-any.whl", hash = "sha256:32e43d604bbe7896fe7c248a9c2276447dbef840feb28fe20494f62af110211d"}, + {file = "cfgv-3.2.0.tar.gz", hash = "sha256:cf22deb93d4bcf92f345a5c3cd39d3d41d6340adc60c78bbbd6588c384fda6a1"}, +] click = [ {file = "click-7.1.2-py2.py3-none-any.whl", hash = "sha256:dacca89f4bfadd5de3d7489b7c8a566eee0d3676333fbb50030263894c38c0dc"}, {file = "click-7.1.2.tar.gz", hash = "sha256:d2b5255c7c6349bc1bd1e59e08cd12acbbd63ce649f2588755783aa94dfb6b1a"}, @@ -1032,6 +1079,10 @@ gitpython = [ {file = "GitPython-3.1.7-py3-none-any.whl", hash = "sha256:fa3b92da728a457dd75d62bb5f3eb2816d99a7fe6c67398e260637a40e3fafb5"}, {file = "GitPython-3.1.7.tar.gz", hash = "sha256:2db287d71a284e22e5c2846042d0602465c7434d910406990d5b74df4afb0858"}, ] +identify = [ + {file = "identify-1.4.25-py2.py3-none-any.whl", hash = "sha256:ccd88716b890ecbe10920659450a635d2d25de499b9a638525a48b48261d989b"}, + {file = "identify-1.4.25.tar.gz", hash = "sha256:110ed090fec6bce1aabe3c72d9258a9de82207adeaa5a05cd75c635880312f9a"}, +] iniconfig = [ {file = "iniconfig-1.0.1-py3-none-any.whl", hash = "sha256:80cf40c597eb564e86346103f609d74efce0f6b4d4f30ec8ce9e2c26411ba437"}, {file = "iniconfig-1.0.1.tar.gz", hash = "sha256:e5f92f89355a67de0595932a6c6c02ab4afddc6fcdc0bfc5becd0d60884d3f69"}, @@ -1091,6 +1142,9 @@ mypy-extensions = [ {file = "mypy_extensions-0.4.3-py2.py3-none-any.whl", hash = "sha256:090fedd75945a69ae91ce1303b5824f428daf5a028d2f6ab8a299250a846f15d"}, {file = "mypy_extensions-0.4.3.tar.gz", hash = "sha256:2d82818f5bb3e369420cb3c4060a7970edba416647068eb4c5343488a6c604a8"}, ] +nodeenv = [ + {file = "nodeenv-1.4.0-py2.py3-none-any.whl", hash = "sha256:4b0b77afa3ba9b54f4b6396e60b0c83f59eaeb2d63dc3cc7a70f7f4af96c82bc"}, +] nox = [ {file = "nox-2020.5.24-py3-none-any.whl", hash = "sha256:c4509621fead99473a1401870e680b0aadadce5c88440f0532863595176d64c1"}, {file = "nox-2020.5.24.tar.gz", hash = "sha256:61a55705736a1a73efbd18d5b262a43d55a1176546e0eb28b29064cfcffe26c0"}, @@ -1115,6 +1169,10 @@ pluggy = [ {file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"}, {file = "pluggy-0.13.1.tar.gz", hash = "sha256:15b2acde666561e1298d71b523007ed7364de07029219b604cf808bfa1c765b0"}, ] +pre-commit = [ + {file = "pre_commit-2.6.0-py2.py3-none-any.whl", hash = "sha256:e8b1315c585052e729ab7e99dcca5698266bedce9067d21dc909c23e3ceed626"}, + {file = "pre_commit-2.6.0.tar.gz", hash = "sha256:1657663fdd63a321a4a739915d7d03baedd555b25054449090f97bb0cb30a915"}, +] py = [ {file = "py-1.9.0-py2.py3-none-any.whl", hash = "sha256:366389d1db726cd2fcfc79732e75410e5fe4d31db13692115529d34069a043c2"}, {file = "py-1.9.0.tar.gz", hash = "sha256:9ca6883ce56b4e8da7e79ac18787889fa5206c79dcc67fb065376cd2fe03f342"}, diff --git a/pyproject.toml b/pyproject.toml index c4934e0..ebc9618 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -30,6 +30,7 @@ python = "^3.8" [tool.poetry.dev-dependencies] # Task Runners nox = "^2020.5.24" +pre-commit = "^2.6.0" # Code Formatters autoflake = "^1.3.1" From 97d714d9eed461fea4b6a612b2ebb1116cd2bbcf Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Tue, 4 Aug 2020 21:14:40 +0200 Subject: [PATCH 06/13] Add CLI entry point `umd` - add the following file: + src/urban_meal_delivery/console.py => click-based CLI tools + tests/test_console.py => tests for the module above - add a CLI entry point `umd`: + implement the --version / -V option to show the installed package's version + rework to package's top-level: * add a __pkg_name__ variable to parameterize the package name + add unit and integration tests - fix that pylint cannot know the proper order of imports in the isolated nox session --- noxfile.py | 15 ++-- poetry.lock | 10 +-- pyproject.toml | 5 ++ setup.cfg | 6 +- src/urban_meal_delivery/__init__.py | 12 ++- src/urban_meal_delivery/console.py | 37 ++++++++ tests/test_console.py | 126 ++++++++++++++++++++++++++++ 7 files changed, 196 insertions(+), 15 deletions(-) create mode 100644 src/urban_meal_delivery/console.py create mode 100644 tests/test_console.py diff --git a/noxfile.py b/noxfile.py index a1a09d6..7204f1a 100644 --- a/noxfile.py +++ b/noxfile.py @@ -96,13 +96,16 @@ def lint(session: Session) -> None: session.run('mypy', '--version') session.run('mypy', *locations) # Ignore errors where pylint cannot import a third-party package due its - # being run in an isolated environment. One way to fix this is to install - # all develop dependencies in this nox session, which we do not do. The - # whole point of static linting tools is to not rely on any package be - # importable at runtime. Instead, these imports are validated implicitly - # when the test suite is run. + # being run in an isolated environment. For the same reason, pylint is + # also not able to determine the correct order of imports. + # One way to fix this is to install all develop dependencies in this nox + # session, which we do not do. The whole point of static linting tools is + # to not rely on any package be importable at runtime. Instead, these + # imports are validated implicitly when the test suite is run. session.run('pylint', '--version') - session.run('pylint', '--disable=import-error', *locations) + session.run( + 'pylint', '--disable=import-error', '--disable=wrong-import-order', *locations, + ) @nox.session(python=[MAIN_PYTHON, NEXT_PYTHON]) diff --git a/poetry.lock b/poetry.lock index 6e55d75..d4b0b02 100644 --- a/poetry.lock +++ b/poetry.lock @@ -127,7 +127,7 @@ python-versions = ">=3.6.1" version = "3.2.0" [[package]] -category = "dev" +category = "main" description = "Composable command line interface toolkit" name = "click" optional = false @@ -836,7 +836,7 @@ description = "Virtual Python Environment builder" name = "virtualenv" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,>=2.7" -version = "20.0.29" +version = "20.0.30" [package.dependencies] appdirs = ">=1.4.3,<2" @@ -886,7 +886,7 @@ python-versions = "*" version = "1.12.1" [metadata] -content-hash = "7a843263817a908ca01d198402d3e9310c33307ac85085f028c8fbdd7587f48f" +content-hash = "899f376a54c187e41392fb83831a59926668b662bfb32004f4a0964c1202698c" lock-version = "1.0" python-versions = "^3.8" @@ -1301,8 +1301,8 @@ typing-extensions = [ {file = "typing_extensions-3.7.4.2.tar.gz", hash = "sha256:79ee589a3caca649a9bfd2a8de4709837400dfa00b6cc81962a1e6a1815969ae"}, ] virtualenv = [ - {file = "virtualenv-20.0.29-py2.py3-none-any.whl", hash = "sha256:8aa9c37b082664dbce2236fa420759c02d64109d8e6013593ad13914718a30fd"}, - {file = "virtualenv-20.0.29.tar.gz", hash = "sha256:f14a0a98ea4397f0d926cff950361766b6a73cd5975ae7eb259d12919f819a25"}, + {file = "virtualenv-20.0.30-py2.py3-none-any.whl", hash = "sha256:8cd7b2a4850b003a11be2fc213e206419efab41115cc14bca20e69654f2ac08e"}, + {file = "virtualenv-20.0.30.tar.gz", hash = "sha256:7b54fd606a1b85f83de49ad8d80dbec08e983a2d2f96685045b262ebc7481ee5"}, ] wemake-python-styleguide = [ {file = "wemake-python-styleguide-0.14.1.tar.gz", hash = "sha256:e13dc580fa56b7b548de8da170bccb8ddff2d4ab026ca987db8a9893bf8a7b5b"}, diff --git a/pyproject.toml b/pyproject.toml index ebc9618..5ee058e 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -27,6 +27,8 @@ repository = "https://github.com/webartifex/urban-meal-delivery" [tool.poetry.dependencies] python = "^3.8" +click = "^7.1.2" + [tool.poetry.dev-dependencies] # Task Runners nox = "^2020.5.24" @@ -51,3 +53,6 @@ wemake-python-styleguide = "^0.14.1" # flake8 plug-in packaging = "^20.4" # used to test the packaged version pytest = "^6.0.1" pytest-cov = "^2.10.0" + +[tool.poetry.scripts] +umd = "urban_meal_delivery.console:main" diff --git a/setup.cfg b/setup.cfg index 59af783..9099cbc 100644 --- a/setup.cfg +++ b/setup.cfg @@ -107,6 +107,8 @@ per-file-ignores = S101, # Shadowing outer scopes occurs naturally with mocks. WPS442, + # No overuse of string constants (e.g., '__version__'). + WPS226, # Explicitly set mccabe's maximum complexity to 10 as recommended by # Thomas McCabe, the inventor of the McCabe complexity, and the NIST. @@ -124,7 +126,9 @@ show-source = true # wemake-python-styleguide's settings # =================================== allowed-domain-names = + param, result, + value, min-name-length = 3 max-name-length = 40 # darglint @@ -173,7 +177,7 @@ single_line_exclusions = typing [mypy] cache_dir = .cache/mypy -[mypy-nox.*,packaging.*,pytest] +[mypy-nox.*,packaging.*,pytest,_pytest.*] ignore_missing_imports = true diff --git a/src/urban_meal_delivery/__init__.py b/src/urban_meal_delivery/__init__.py index d17278b..5a68f43 100644 --- a/src/urban_meal_delivery/__init__.py +++ b/src/urban_meal_delivery/__init__.py @@ -1,9 +1,15 @@ """Source code for the urban-meal-delivery research project.""" -from importlib import metadata +from importlib import metadata as _metadata try: - __version__ = metadata.version(__name__) -except metadata.PackageNotFoundError: # pragma: no cover + _pkg_info = _metadata.metadata(__name__) + +except _metadata.PackageNotFoundError: # pragma: no cover + __pkg_name__ = 'unknown' __version__ = 'unknown' + +else: + __pkg_name__ = _pkg_info['name'] + __version__ = _pkg_info['version'] diff --git a/src/urban_meal_delivery/console.py b/src/urban_meal_delivery/console.py new file mode 100644 index 0000000..0141370 --- /dev/null +++ b/src/urban_meal_delivery/console.py @@ -0,0 +1,37 @@ +"""Provide CLI scripts for the project.""" + +from typing import Any + +import click +from click.core import Context + +import urban_meal_delivery + + +def show_version(ctx: Context, _param: Any, value: bool) -> None: + """Show the package's version.""" + # If --version / -V is NOT passed in, + # continue with the command. + if not value or ctx.resilient_parsing: + return + + # Mimic the colors of `poetry version`. + pkg_name = click.style(urban_meal_delivery.__pkg_name__, fg='green') # noqa:WPS609 + version = click.style(urban_meal_delivery.__version__, fg='blue') # noqa:WPS609 + # Include a warning for development versions. + warning = click.style(' (development)', fg='red') if '.dev' in version else '' + click.echo(f'{pkg_name}, version {version}{warning}') + ctx.exit() + + +@click.command() +@click.option( + '--version', + '-V', + is_flag=True, + callback=show_version, + is_eager=True, + expose_value=False, +) +def main() -> None: + """The urban-meal-delivery research project.""" diff --git a/tests/test_console.py b/tests/test_console.py new file mode 100644 index 0000000..0b3cba8 --- /dev/null +++ b/tests/test_console.py @@ -0,0 +1,126 @@ +"""Test the package's `umd` command-line client.""" + +import click +import pytest +from _pytest.capture import CaptureFixture # noqa:WPS436 +from _pytest.monkeypatch import MonkeyPatch # noqa:WPS436 +from click import testing as click_testing + +from urban_meal_delivery import console + + +@pytest.fixture +def ctx() -> click.Context: + """Context around the console.main Command.""" + return click.Context(console.main) + + +class TestShowVersion: + """Test console.show_version(). + + The function is used as a callback to a click command option. + + show_version() prints the name and version of the installed package to + stdout. The output looks like this: "{pkg_name}, version {version}". + + Development (= non-final) versions are indicated by appending a + " (development)" to the output. + """ + + # pylint:disable=no-self-use + + def test_no_version(self, capsys: CaptureFixture, ctx: click.Context) -> None: + """The the early exit branch without any output.""" + console.show_version(ctx, _param='discarded', value=False) + + captured = capsys.readouterr() + + assert captured.out == '' + + def test_final_version( + self, capsys: CaptureFixture, ctx: click.Context, monkeypatch: MonkeyPatch, + ) -> None: + """For final versions, NO "development" warning is emitted.""" + version = '1.2.3' + monkeypatch.setattr(console.urban_meal_delivery, '__version__', version) + + with pytest.raises(click.exceptions.Exit): + console.show_version(ctx, _param='discarded', value=True) + + captured = capsys.readouterr() + + assert captured.out.endswith(f', version {version}\n') + + def test_develop_version( + self, capsys: CaptureFixture, ctx: click.Context, monkeypatch: MonkeyPatch, + ) -> None: + """For develop versions, a warning thereof is emitted.""" + version = '1.2.3.dev0' + monkeypatch.setattr(console.urban_meal_delivery, '__version__', version) + + with pytest.raises(click.exceptions.Exit): + console.show_version(ctx, _param='discarded', value=True) + + captured = capsys.readouterr() + + assert captured.out.strip().endswith(f', version {version} (development)') + + +@pytest.fixture +def cli() -> click_testing.CliRunner: + """Initialize Click's CLI Test Runner.""" + return click_testing.CliRunner() + + +class TestCLI: + """Test the `umd` CLI utility. + + The test cases are integration tests. + Therefore, they are not considered for coverage reporting. + """ + + # pylint:disable=no-self-use + + @pytest.mark.no_cover + def test_no_options(self, cli: click_testing.CliRunner) -> None: + """Exit with 0 status code and no output if run without options.""" + result = cli.invoke(console.main) + + assert result.exit_code == 0 + assert result.output == '' + + # The following test cases validate the --version / -V option. + + version_options = ('--version', '-V') + + @pytest.mark.no_cover + @pytest.mark.parametrize( + 'option', version_options, + ) + def test_final_version( + self, cli: click_testing.CliRunner, monkeypatch: MonkeyPatch, option: str, + ) -> None: + """For final versions, NO "development" warning is emitted.""" + version = '1.2.3' + monkeypatch.setattr(console.urban_meal_delivery, '__version__', version) + + result = cli.invoke(console.main, option) + + assert result.exit_code == 0 + assert result.output.strip().endswith(f', version {version}') + + @pytest.mark.no_cover + @pytest.mark.parametrize( + 'option', version_options, + ) + def test_develop_version( + self, cli: click_testing.CliRunner, monkeypatch: MonkeyPatch, option: str, + ) -> None: + """For develop versions, a warning thereof is emitted.""" + version = '1.2.3.dev0' + monkeypatch.setattr(console.urban_meal_delivery, '__version__', version) + + result = cli.invoke(console.main, option) + + assert result.exit_code == 0 + assert result.output.strip().endswith(f', version {version} (development)') From 8586db58c71a1e1fd86ee98c52e91466453a4a44 Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Tue, 4 Aug 2020 22:57:55 +0200 Subject: [PATCH 07/13] Run type checks only against packaged *.py files - for tests/ and the noxfile.py, type annotations are not strictly enforced any more + this simplifies the way test cases and nox sessions are written + for many pytest fixtures, no types are available via a public API - put fixtures inside the classes the corresponding test cases are grouped in --- noxfile.py | 37 +++++++++++++++++------------- setup.cfg | 6 ++++- tests/test_console.py | 52 +++++++++++++++---------------------------- tests/test_version.py | 26 ++++++++++------------ 4 files changed, 57 insertions(+), 64 deletions(-) diff --git a/noxfile.py b/noxfile.py index 7204f1a..b743fb8 100644 --- a/noxfile.py +++ b/noxfile.py @@ -3,7 +3,6 @@ import contextlib import os import tempfile -from typing import Any, Generator import nox from nox.sessions import Session @@ -14,11 +13,14 @@ MAIN_PYTHON = '3.8' # Keep the project is forward compatible. NEXT_PYTHON = '3.9' +# Path to the *.py files to be packaged. +PACKAGE_SOURCE_LOCATION = 'src/' + # Path to the test suite. PYTEST_LOCATION = 'tests/' -# Paths with *.py files. -SRC_LOCATIONS = 'noxfile.py', 'src/', PYTEST_LOCATION +# Paths with all *.py files. +SRC_LOCATIONS = 'noxfile.py', PACKAGE_SOURCE_LOCATION, PYTEST_LOCATION # Use a unified .cache/ folder for all tools. @@ -38,7 +40,7 @@ nox.options.sessions = ( @nox.session(name='format', python=MAIN_PYTHON) -def format_(session: Session) -> None: +def format_(session): """Format source files with autoflake, black, and isort. If no extra arguments are provided, all source files are formatted. @@ -68,7 +70,7 @@ def format_(session: Session) -> None: @nox.session(python=MAIN_PYTHON) -def lint(session: Session) -> None: +def lint(session): """Lint source files with flake8, mypy, and pylint. If no extra arguments are provided, all source files are linted. @@ -93,8 +95,15 @@ def lint(session: Session) -> None: with _isort_fix(session): # TODO (isort): Remove after upgrading session.run('isort', '--version') session.run('isort', '--check-only', *locations) - session.run('mypy', '--version') - session.run('mypy', *locations) + # For mypy, only lint *.py files to be packaged. + mypy_locations = [ + path for path in locations if path.startswith(PACKAGE_SOURCE_LOCATION) + ] + if mypy_locations: + session.run('mypy', '--version') + session.run('mypy', *mypy_locations) + else: + session.log('No paths to be checked with mypy') # Ignore errors where pylint cannot import a third-party package due its # being run in an isolated environment. For the same reason, pylint is # also not able to determine the correct order of imports. @@ -109,7 +118,7 @@ def lint(session: Session) -> None: @nox.session(python=[MAIN_PYTHON, NEXT_PYTHON]) -def test(session: Session) -> None: +def test(session): """Test the code base. Runs the unit and integration tests (written with pytest). @@ -148,7 +157,7 @@ def test(session: Session) -> None: @nox.session(name='pre-commit', python=MAIN_PYTHON, venv_backend='none') -def pre_commit(session: Session) -> None: +def pre_commit(session): """Source files must be well-formed before they enter git. Intended to be run as a pre-commit hook. @@ -165,7 +174,7 @@ def pre_commit(session: Session) -> None: @nox.session(name='pre-merge', python=MAIN_PYTHON) -def pre_merge(session: Session) -> None: +def pre_merge(session): """The test suite must pass before merges are made. Intended to be run either as a pre-merge or pre-push hook. @@ -188,7 +197,7 @@ def pre_merge(session: Session) -> None: test(session) -def _begin(session: Session) -> None: +def _begin(session): """Show generic info about a session.""" if session.posargs: # Part of the hack in pre_merge() to "drop" the extra arguments. @@ -205,9 +214,7 @@ def _begin(session: Session) -> None: print(os.getcwd()) # noqa:WPS421 -def _install_packages( - session: Session, *packages_or_pip_args: str, **kwargs: Any, -) -> None: +def _install_packages(session: Session, *packages_or_pip_args: str, **kwargs) -> None: """Install packages respecting the poetry.lock file. This function wraps nox.sessions.Session.install() such that it installs @@ -251,7 +258,7 @@ def _install_packages( # TODO (isort): Remove this fix after # upgrading to isort ^5.2.2 in pyproject.toml. @contextlib.contextmanager -def _isort_fix(session: Session) -> Generator: +def _isort_fix(session): """Temporarily upgrade to isort 5.2.2.""" session.install('isort==5.2.2') try: diff --git a/setup.cfg b/setup.cfg index 9099cbc..f275247 100644 --- a/setup.cfg +++ b/setup.cfg @@ -96,6 +96,8 @@ extend-ignore = per-file-ignores = noxfile.py: + # Type annotations are not strictly enforced. + ANN0, ANN2, # TODO (isort): Check if still too many module members. WPS202, # TODO (isort): Remove after simplifying the nox session "lint". @@ -103,6 +105,8 @@ per-file-ignores = # No overuse of string constants (e.g., '--version'). WPS226, tests/*.py: + # Type annotations are not strictly enforced. + ANN0, ANN2, # `assert` statements are ok in the test suite. S101, # Shadowing outer scopes occurs naturally with mocks. @@ -177,7 +181,7 @@ single_line_exclusions = typing [mypy] cache_dir = .cache/mypy -[mypy-nox.*,packaging.*,pytest,_pytest.*] +[mypy-nox.*,packaging,pytest] ignore_missing_imports = true diff --git a/tests/test_console.py b/tests/test_console.py index 0b3cba8..00c721f 100644 --- a/tests/test_console.py +++ b/tests/test_console.py @@ -2,19 +2,11 @@ import click import pytest -from _pytest.capture import CaptureFixture # noqa:WPS436 -from _pytest.monkeypatch import MonkeyPatch # noqa:WPS436 from click import testing as click_testing from urban_meal_delivery import console -@pytest.fixture -def ctx() -> click.Context: - """Context around the console.main Command.""" - return click.Context(console.main) - - class TestShowVersion: """Test console.show_version(). @@ -29,7 +21,12 @@ class TestShowVersion: # pylint:disable=no-self-use - def test_no_version(self, capsys: CaptureFixture, ctx: click.Context) -> None: + @pytest.fixture + def ctx(self) -> click.Context: + """Context around the console.main Command.""" + return click.Context(console.main) + + def test_no_version(self, capsys, ctx): """The the early exit branch without any output.""" console.show_version(ctx, _param='discarded', value=False) @@ -37,9 +34,7 @@ class TestShowVersion: assert captured.out == '' - def test_final_version( - self, capsys: CaptureFixture, ctx: click.Context, monkeypatch: MonkeyPatch, - ) -> None: + def test_final_version(self, capsys, ctx, monkeypatch): """For final versions, NO "development" warning is emitted.""" version = '1.2.3' monkeypatch.setattr(console.urban_meal_delivery, '__version__', version) @@ -51,9 +46,7 @@ class TestShowVersion: assert captured.out.endswith(f', version {version}\n') - def test_develop_version( - self, capsys: CaptureFixture, ctx: click.Context, monkeypatch: MonkeyPatch, - ) -> None: + def test_develop_version(self, capsys, ctx, monkeypatch): """For develop versions, a warning thereof is emitted.""" version = '1.2.3.dev0' monkeypatch.setattr(console.urban_meal_delivery, '__version__', version) @@ -66,12 +59,6 @@ class TestShowVersion: assert captured.out.strip().endswith(f', version {version} (development)') -@pytest.fixture -def cli() -> click_testing.CliRunner: - """Initialize Click's CLI Test Runner.""" - return click_testing.CliRunner() - - class TestCLI: """Test the `umd` CLI utility. @@ -81,8 +68,13 @@ class TestCLI: # pylint:disable=no-self-use + @pytest.fixture + def cli(self) -> click_testing.CliRunner: + """Initialize Click's CLI Test Runner.""" + return click_testing.CliRunner() + @pytest.mark.no_cover - def test_no_options(self, cli: click_testing.CliRunner) -> None: + def test_no_options(self, cli): """Exit with 0 status code and no output if run without options.""" result = cli.invoke(console.main) @@ -94,12 +86,8 @@ class TestCLI: version_options = ('--version', '-V') @pytest.mark.no_cover - @pytest.mark.parametrize( - 'option', version_options, - ) - def test_final_version( - self, cli: click_testing.CliRunner, monkeypatch: MonkeyPatch, option: str, - ) -> None: + @pytest.mark.parametrize('option', version_options) + def test_final_version(self, cli, monkeypatch, option): """For final versions, NO "development" warning is emitted.""" version = '1.2.3' monkeypatch.setattr(console.urban_meal_delivery, '__version__', version) @@ -110,12 +98,8 @@ class TestCLI: assert result.output.strip().endswith(f', version {version}') @pytest.mark.no_cover - @pytest.mark.parametrize( - 'option', version_options, - ) - def test_develop_version( - self, cli: click_testing.CliRunner, monkeypatch: MonkeyPatch, option: str, - ) -> None: + @pytest.mark.parametrize('option', version_options) + def test_develop_version(self, cli, monkeypatch, option): """For develop versions, a warning thereof is emitted.""" version = '1.2.3.dev0' monkeypatch.setattr(console.urban_meal_delivery, '__version__', version) diff --git a/tests/test_version.py b/tests/test_version.py index e3bf23c..474b7b1 100644 --- a/tests/test_version.py +++ b/tests/test_version.py @@ -13,35 +13,33 @@ import re import pytest from packaging import version as pkg_version -from packaging.version import Version import urban_meal_delivery -@pytest.fixture -def parsed_version() -> str: - """The packaged version.""" - return pkg_version.Version(urban_meal_delivery.__version__) # noqa:WPS609 - - class TestPEP404Compliance: """Packaged version identifier is PEP440 compliant.""" # pylint:disable=no-self-use - def test_parsed_version_has_no_epoch(self, parsed_version: Version) -> None: + @pytest.fixture + def parsed_version(self) -> str: + """The packaged version.""" + return pkg_version.Version(urban_meal_delivery.__version__) # noqa:WPS609 + + def test_parsed_version_has_no_epoch(self, parsed_version): """PEP440 compliant subset of semantic versioning: no epoch.""" assert parsed_version.epoch == 0 - def test_parsed_version_is_non_local(self, parsed_version: Version) -> None: + def test_parsed_version_is_non_local(self, parsed_version): """PEP440 compliant subset of semantic versioning: no local version.""" assert parsed_version.local is None - def test_parsed_version_is_no_post_release(self, parsed_version: Version) -> None: + def test_parsed_version_is_no_post_release(self, parsed_version): """PEP440 compliant subset of semantic versioning: no post releases.""" assert parsed_version.is_postrelease is False - def test_parsed_version_is_all_public(self, parsed_version: Version) -> None: + def test_parsed_version_is_all_public(self, parsed_version): """PEP440 compliant subset of semantic versioning: all public parts.""" assert parsed_version.public == urban_meal_delivery.__version__ # noqa:WPS609 @@ -55,7 +53,7 @@ class TestSemanticVersioning: r'^(0|([1-9]\d*))\.(0|([1-9]\d*))\.(0|([1-9]\d*))(\.dev(0|([1-9]\d*)))?$', ) - def test_version_is_semantic(self) -> None: + def test_version_is_semantic(self): """Packaged version follows semantic versioning.""" result = self.version_pattern.fullmatch( urban_meal_delivery.__version__, # noqa:WPS609 @@ -79,7 +77,7 @@ class TestSemanticVersioning: '1.2.3.dev10', ], ) - def test_valid_semantic_versioning(self, version: str) -> None: + def test_valid_semantic_versioning(self, version): """Versions follow the x.y.z or x.y.z.devN format.""" result = self.version_pattern.fullmatch(version) @@ -109,7 +107,7 @@ class TestSemanticVersioning: '1.2..3', ], ) - def test_invalid_semantic_versioning(self, version: str) -> None: + def test_invalid_semantic_versioning(self, version): """Versions follow the x.y.z or x.y.z.devN format.""" result = self.version_pattern.fullmatch(version) From 48fe2f68790064b7361dbc1584e58e0e203f214a Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Tue, 4 Aug 2020 23:16:15 +0200 Subject: [PATCH 08/13] Add security checks for the dependencies - add a nox session "safety" --- noxfile.py | 22 ++++++++++++++++++++++ 1 file changed, 22 insertions(+) diff --git a/noxfile.py b/noxfile.py index b743fb8..8fdfc1f 100644 --- a/noxfile.py +++ b/noxfile.py @@ -36,6 +36,7 @@ nox.options.sessions = ( 'lint', f'test-{MAIN_PYTHON}', f'test-{NEXT_PYTHON}', + 'safety', ) @@ -197,6 +198,27 @@ def pre_merge(session): test(session) +@nox.session(python=MAIN_PYTHON) +def safety(session): + """Check the dependencies for known security vulnerabilities.""" + _begin(session) + # We do not pin the version of `safety` to always check with + # the latest version. The risk this breaks the CI is rather low. + session.install('safety') + with tempfile.NamedTemporaryFile() as requirements_txt: + session.run( + 'poetry', + 'export', + '--dev', + '--format=requirements.txt', + f'--output={requirements_txt.name}', + external=True, + ) + session.run( + 'safety', 'check', f'--file={requirements_txt.name}', '--full-report', + ) + + def _begin(session): """Show generic info about a session.""" if session.posargs: From 126dcf7c39b95a756b7a31c2112aee59fe795579 Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Wed, 5 Aug 2020 00:02:40 +0200 Subject: [PATCH 09/13] Include doctests in the test suite - use xdoctest to validate all code snippets in docstrings - add xdoctest to the nox session "test" --- noxfile.py | 24 ++++++++++++++++++---- poetry.lock | 32 +++++++++++++++++++++++++++-- pyproject.toml | 1 + src/urban_meal_delivery/__init__.py | 8 +++++++- 4 files changed, 58 insertions(+), 7 deletions(-) diff --git a/noxfile.py b/noxfile.py index 8fdfc1f..f60687d 100644 --- a/noxfile.py +++ b/noxfile.py @@ -13,6 +13,8 @@ MAIN_PYTHON = '3.8' # Keep the project is forward compatible. NEXT_PYTHON = '3.9' +PACKAGE_IMPORT_NAME = 'urban_meal_delivery' + # Path to the *.py files to be packaged. PACKAGE_SOURCE_LOCATION = 'src/' @@ -48,6 +50,8 @@ def format_(session): Otherwise, they are interpreted as paths the formatters work on recursively. """ _begin(session) + # The formatting tools do not require the developed + # package be installed in the virtual environment. _install_packages(session, 'autoflake', 'black', 'isort') # Interpret extra arguments as locations of source files. locations = session.posargs or SRC_LOCATIONS @@ -78,6 +82,8 @@ def lint(session): Otherwise, they are interpreted as paths the linters work on recursively. """ _begin(session) + # The linting tools do not require the developed + # package be installed in the virtual environment. _install_packages( session, 'flake8', @@ -122,13 +128,15 @@ def lint(session): def test(session): """Test the code base. - Runs the unit and integration tests (written with pytest). + Runs the unit and integration tests with pytest and + validate that all code snippets in docstrings work with xdoctest. If no extra arguments are provided, the entire test suite is exexcuted and succeeds only with 100% coverage. If extra arguments are provided, they are - forwarded to pytest without any changes. + forwarded to pytest and xdoctest without any changes. + xdoctest ignores arguments it does not understand. """ # Re-using an old environment is not so easy here as # `poetry install --no-dev` removes previously installed packages. @@ -139,9 +147,12 @@ def test(session): ) _begin(session) - # Install only the non-develop dependencies and the testing tool chain. + # The testing tools require the developed package and its + # non-develop dependencies be installed in the virtual environment. session.run('poetry', 'install', '--no-dev', external=True) - _install_packages(session, 'pytest', 'pytest-cov') + _install_packages( + session, 'packaging', 'pytest', 'pytest-cov', 'xdoctest[optional]', + ) # Interpret extra arguments as options for pytest. # They are "dropped" by the hack in the pre_merge() function # if this function is run within the "pre-merge" session. @@ -154,7 +165,12 @@ def test(session): '--cov-report=term-missing:skip-covered', PYTEST_LOCATION, ) + session.run('pytest', '--version') session.run('pytest', *args) + # For xdoctest, the default arguments are different from pytest. + args = posargs or [PACKAGE_IMPORT_NAME] + session.run('xdoctest', '--version') + session.run('xdoctest', '--quiet', *args) # --quiet => less verbose output @nox.session(name='pre-commit', python=MAIN_PYTHON, venv_backend='none') diff --git a/poetry.lock b/poetry.lock index d4b0b02..78a265a 100644 --- a/poetry.lock +++ b/poetry.lock @@ -137,7 +137,7 @@ version = "7.1.2" [[package]] category = "dev" description = "Cross-platform colored terminal text." -marker = "sys_platform == \"win32\" or platform_system == \"Windows\"" +marker = "sys_platform == \"win32\" or platform_system == \"Windows\" or platform_system == \"Windows\"" name = "colorama" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" @@ -885,8 +885,32 @@ optional = false python-versions = "*" version = "1.12.1" +[[package]] +category = "dev" +description = "A rewrite of the builtin doctest module" +name = "xdoctest" +optional = false +python-versions = "*" +version = "0.13.0" + +[package.dependencies] +six = "*" + +[package.dependencies.Pygments] +optional = true +version = "*" + +[package.dependencies.colorama] +optional = true +version = "*" + +[package.extras] +all = ["six", "pytest", "pytest-cov", "codecov", "scikit-build", "cmake", "ninja", "pybind11", "pygments", "colorama"] +optional = ["pygments", "colorama"] +tests = ["pytest", "pytest-cov", "codecov", "scikit-build", "cmake", "ninja", "pybind11"] + [metadata] -content-hash = "899f376a54c187e41392fb83831a59926668b662bfb32004f4a0964c1202698c" +content-hash = "79e46e67260312ba1b227f7717d0ba781a6ad5df352f83fe6e1b72a98ad1ec5b" lock-version = "1.0" python-versions = "^3.8" @@ -1311,3 +1335,7 @@ wemake-python-styleguide = [ wrapt = [ {file = "wrapt-1.12.1.tar.gz", hash = "sha256:b62ffa81fb85f4332a4f609cab4ac40709470da05643a082ec1eb88e6d9b97d7"}, ] +xdoctest = [ + {file = "xdoctest-0.13.0-py2.py3-none-any.whl", hash = "sha256:de861fd5230a46bd26c054b4981169dd963f813768cb62b62e104e4d2644ac94"}, + {file = "xdoctest-0.13.0.tar.gz", hash = "sha256:4f113a430076561a9d7f31af65b5d5acda62ee06b05cb6894264cb9efb8196ac"}, +] diff --git a/pyproject.toml b/pyproject.toml index 5ee058e..200a42a 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -53,6 +53,7 @@ wemake-python-styleguide = "^0.14.1" # flake8 plug-in packaging = "^20.4" # used to test the packaged version pytest = "^6.0.1" pytest-cov = "^2.10.0" +xdoctest = { version="^0.13.0", extras=["optional"] } [tool.poetry.scripts] umd = "urban_meal_delivery.console:main" diff --git a/src/urban_meal_delivery/__init__.py b/src/urban_meal_delivery/__init__.py index 5a68f43..425f95f 100644 --- a/src/urban_meal_delivery/__init__.py +++ b/src/urban_meal_delivery/__init__.py @@ -1,4 +1,10 @@ -"""Source code for the urban-meal-delivery research project.""" +"""Source code for the urban-meal-delivery research project. + +Example: + >>> import urban_meal_delivery as umd + >>> umd.__version__ != '0.0.0' + True +""" from importlib import metadata as _metadata From 882226f0a925dd495661c9b6f366171c6e25eb1d Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Wed, 5 Aug 2020 01:34:27 +0200 Subject: [PATCH 10/13] Add technical documentation for the package - use sphinx to document the developed package - create a nox session "docs" that builds the docs - include a skeleton in the docs/ folder + how to install the package + how to use nox + license --- docs/conf.py | 15 ++ docs/index.rst | 95 ++++++++ docs/license.rst | 4 + docs/reference.rst | 6 + noxfile.py | 36 ++- poetry.lock | 339 +++++++++++++++++++++++++++- pyproject.toml | 4 + setup.cfg | 5 + src/urban_meal_delivery/__init__.py | 2 + 9 files changed, 503 insertions(+), 3 deletions(-) create mode 100644 docs/conf.py create mode 100644 docs/index.rst create mode 100644 docs/license.rst create mode 100644 docs/reference.rst diff --git a/docs/conf.py b/docs/conf.py new file mode 100644 index 0000000..2ab87a7 --- /dev/null +++ b/docs/conf.py @@ -0,0 +1,15 @@ +"""Configure sphinx.""" + +import urban_meal_delivery as umd + + +project = umd.__pkg_name__ +author = umd.__author__ +copyright = f'2020, {author}' # pylint:disable=redefined-builtin +version = release = umd.__version__ + +extensions = [ + 'sphinx.ext.autodoc', + 'sphinx.ext.napoleon', + 'sphinx_autodoc_typehints', +] diff --git a/docs/index.rst b/docs/index.rst new file mode 100644 index 0000000..87b8f19 --- /dev/null +++ b/docs/index.rst @@ -0,0 +1,95 @@ +Urban Meal Delivery +=================== + +.. toctree:: + :hidden: + :maxdepth: 1 + + license + reference + + +This is the documentation for the `urban-meal-delivery` package +that contains all the source code +used in a research project +residing in this `repository`_. + + +Prerequisites +------------- + +Only `git`_, `Python`_ 3.8 and `poetry`_ must be available. +All other software is automatically installed (cf., next section). + + +Installation +------------ + +The `urban-meal-delivery` package is installed +only after cloning the `repository`_. +It is *not* available on `PyPI`_ via `pip`_. + +.. code-block:: console + + $ git clone https://github.com/webartifex/urban-meal-delivery.git + +Use `poetry`_ to install the package +with its dependencies into a `virtual environment`_, +which we also refer to as the **local** or the **develop** environment here. + +.. code-block:: console + + $ poetry install + + +First Steps +----------- + +Verify that the installation is in a good state by +running the test suite and the code linters +via the automated task runner `nox`_. + +``poetry run`` executes whatever comes after it in the develop environment. +If you have a project-independent `nox`_ installation available, +you may drop the ``poetry run`` prefix. +Otherwise, `nox`_ was installed as a develop dependency in the `virtual environment`_ +in the installation step above. + +Run the default sessions in `nox`_: + +.. code-block:: console + + $ [poetry run] nox + +`nox`_ provides many options. +You can use them to choose among the different tasks you want to achieve. + +.. option:: -l, --list + + List all available `nox`_ sessions. + This includes sessions that are *not* run by default + and that may be used as develop tools + to work on the source files. + +.. option:: -s [SESSION [SESSION ...]], --session[s] [SESSION [SESSION ...]] + + Run only the specified session[s] + +.. option:: -r, --reuse-existing-virtualenvs + + By default, + `nox`_ creates a new `virtual environment`_ for every session. + To speed things up, + one can re-use existing environments with the -r flag. + This should only be done when developing + and not before committing or on the CI server. + + +.. _git: https://git-scm.com/ +.. _nox: https://nox.thea.codes/en/stable/ +.. _pip: https://pip.pypa.io/en/stable/ +.. _poetry: https://python-poetry.org/docs/ +.. _pypi: https://pypi.org/ +.. _python: https://docs.python.org/3/ +.. _repository: https://github.com/webartifex/urban-meal-delivery +.. _virtual environment: https://docs.python.org/3/tutorial/venv.html diff --git a/docs/license.rst b/docs/license.rst new file mode 100644 index 0000000..8f93aa7 --- /dev/null +++ b/docs/license.rst @@ -0,0 +1,4 @@ +License +======= + +.. include:: ../LICENSE.txt diff --git a/docs/reference.rst b/docs/reference.rst new file mode 100644 index 0000000..066d730 --- /dev/null +++ b/docs/reference.rst @@ -0,0 +1,6 @@ +Reference +========= + +.. contents:: + :local: + :backlinks: none diff --git a/noxfile.py b/noxfile.py index f60687d..36f7813 100644 --- a/noxfile.py +++ b/noxfile.py @@ -15,6 +15,10 @@ NEXT_PYTHON = '3.9' PACKAGE_IMPORT_NAME = 'urban_meal_delivery' +# Docs/sphinx locations. +DOCS_SRC = 'docs/' +DOCS_BUILD = '.cache/docs/' + # Path to the *.py files to be packaged. PACKAGE_SOURCE_LOCATION = 'src/' @@ -22,8 +26,12 @@ PACKAGE_SOURCE_LOCATION = 'src/' PYTEST_LOCATION = 'tests/' # Paths with all *.py files. -SRC_LOCATIONS = 'noxfile.py', PACKAGE_SOURCE_LOCATION, PYTEST_LOCATION - +SRC_LOCATIONS = ( + f'{DOCS_SRC}/conf.py', + 'noxfile.py', + PACKAGE_SOURCE_LOCATION, + PYTEST_LOCATION, +) # Use a unified .cache/ folder for all tools. nox.options.envdir = '.cache/nox' @@ -39,6 +47,7 @@ nox.options.sessions = ( f'test-{MAIN_PYTHON}', f'test-{NEXT_PYTHON}', 'safety', + 'docs', ) @@ -235,6 +244,29 @@ def safety(session): ) +@nox.session(python=MAIN_PYTHON) +def docs(session): + """Build the documentation with sphinx.""" + # The latest version of the package needs to be installed + # so that sphinx's autodoc can include the latest docstrings. + if session.virtualenv.reuse_existing: + raise RuntimeError('The "docs" session must be run without the "-r" option') + + _begin(session) + + # The documentation tools require the developed package and its + # non-develop dependencies be installed in the virtual environment. + # Otherwise, sphinx's autodoc could not include the docstrings. + session.run('poetry', 'install', '--no-dev', external=True) + _install_packages(session, 'sphinx', 'sphinx-autodoc-typehints') + + session.run('sphinx-build', DOCS_SRC, DOCS_BUILD) + # Verify all external links return 200 OK. + session.run('sphinx-build', '-b', 'linkcheck', DOCS_SRC, DOCS_BUILD) + + print(f'Docs are available at {os.getcwd()}/{DOCS_BUILD}index.html') # noqa:WPS421 + + def _begin(session): """Show generic info about a session.""" if session.posargs: diff --git a/poetry.lock b/poetry.lock index 78a265a..195fcc9 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,3 +1,11 @@ +[[package]] +category = "dev" +description = "A configurable sidebar-enabled Sphinx theme" +name = "alabaster" +optional = false +python-versions = "*" +version = "0.7.12" + [[package]] category = "dev" description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." @@ -83,6 +91,17 @@ version = "1.3.1" [package.dependencies] pyflakes = ">=1.1.0" +[[package]] +category = "dev" +description = "Internationalization utilities" +name = "babel" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "2.8.0" + +[package.dependencies] +pytz = ">=2015.7" + [[package]] category = "dev" description = "Security oriented static analyser for python code." @@ -118,6 +137,14 @@ typed-ast = ">=1.4.0" [package.extras] d = ["aiohttp (>=3.3.2)", "aiohttp-cors"] +[[package]] +category = "dev" +description = "Python package for providing Mozilla's CA Bundle." +name = "certifi" +optional = false +python-versions = "*" +version = "2020.6.20" + [[package]] category = "dev" description = "Validate configuration and produce human readable error messages." @@ -126,6 +153,14 @@ optional = false python-versions = ">=3.6.1" version = "3.2.0" +[[package]] +category = "dev" +description = "Universal encoding detector for Python 2 and 3" +name = "chardet" +optional = false +python-versions = "*" +version = "3.0.4" + [[package]] category = "main" description = "Composable command line interface toolkit" @@ -466,6 +501,22 @@ version = "1.4.25" [package.extras] license = ["editdistance"] +[[package]] +category = "dev" +description = "Internationalized Domain Names in Applications (IDNA)" +name = "idna" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "2.10" + +[[package]] +category = "dev" +description = "Getting image size from png/jpeg/jpeg2000/gif file" +name = "imagesize" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +version = "1.2.0" + [[package]] category = "dev" description = "iniconfig: brain-dead simple config-ini parsing" @@ -488,6 +539,20 @@ pyproject = ["toml"] requirements = ["pipreqs", "pip-api"] xdg_home = ["appdirs (>=1.4.0)"] +[[package]] +category = "dev" +description = "A very fast and expressive template engine." +name = "jinja2" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +version = "2.11.2" + +[package.dependencies] +MarkupSafe = ">=0.23" + +[package.extras] +i18n = ["Babel (>=0.8)"] + [[package]] category = "dev" description = "A fast and thorough lazy object proxy." @@ -496,6 +561,14 @@ optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" version = "1.4.3" +[[package]] +category = "dev" +description = "Safely add untrusted strings to HTML/XML markup." +name = "markupsafe" +optional = false +python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*" +version = "1.1.1" + [[package]] category = "dev" description = "McCabe checker, plugin for flake8" @@ -731,6 +804,14 @@ pytest = ">=4.6" [package.extras] testing = ["fields", "hunter", "process-tests (2.0.2)", "six", "pytest-xdist", "virtualenv"] +[[package]] +category = "dev" +description = "World timezone definitions, modern and historical" +name = "pytz" +optional = false +python-versions = "*" +version = "2020.1" + [[package]] category = "dev" description = "YAML parser and emitter for Python" @@ -747,6 +828,24 @@ optional = false python-versions = "*" version = "2020.7.14" +[[package]] +category = "dev" +description = "Python HTTP for Humans." +name = "requests" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +version = "2.24.0" + +[package.dependencies] +certifi = ">=2017.4.17" +chardet = ">=3.0.2,<4" +idna = ">=2.5,<3" +urllib3 = ">=1.21.1,<1.25.0 || >1.25.0,<1.25.1 || >1.25.1,<1.26" + +[package.extras] +security = ["pyOpenSSL (>=0.14)", "cryptography (>=1.3.4)"] +socks = ["PySocks (>=1.5.6,<1.5.7 || >1.5.7)", "win-inet-pton"] + [[package]] category = "dev" description = "reStructuredText linter" @@ -782,6 +881,124 @@ optional = false python-versions = "*" version = "2.0.0" +[[package]] +category = "dev" +description = "Python documentation generator" +name = "sphinx" +optional = false +python-versions = ">=3.5" +version = "3.1.2" + +[package.dependencies] +Jinja2 = ">=2.3" +Pygments = ">=2.0" +alabaster = ">=0.7,<0.8" +babel = ">=1.3" +colorama = ">=0.3.5" +docutils = ">=0.12" +imagesize = "*" +packaging = "*" +requests = ">=2.5.0" +setuptools = "*" +snowballstemmer = ">=1.1" +sphinxcontrib-applehelp = "*" +sphinxcontrib-devhelp = "*" +sphinxcontrib-htmlhelp = "*" +sphinxcontrib-jsmath = "*" +sphinxcontrib-qthelp = "*" +sphinxcontrib-serializinghtml = "*" + +[package.extras] +docs = ["sphinxcontrib-websupport"] +lint = ["flake8 (>=3.5.0)", "flake8-import-order", "mypy (>=0.780)", "docutils-stubs"] +test = ["pytest", "pytest-cov", "html5lib", "typed-ast", "cython"] + +[[package]] +category = "dev" +description = "Type hints (PEP 484) support for the Sphinx autodoc extension" +name = "sphinx-autodoc-typehints" +optional = false +python-versions = ">=3.5.2" +version = "1.11.0" + +[package.dependencies] +Sphinx = ">=3.0" + +[package.extras] +test = ["pytest (>=3.1.0)", "typing-extensions (>=3.5)", "sphobjinv (>=2.0)", "dataclasses"] +type_comments = ["typed-ast (>=1.4.0)"] + +[[package]] +category = "dev" +description = "sphinxcontrib-applehelp is a sphinx extension which outputs Apple help books" +name = "sphinxcontrib-applehelp" +optional = false +python-versions = ">=3.5" +version = "1.0.2" + +[package.extras] +lint = ["flake8", "mypy", "docutils-stubs"] +test = ["pytest"] + +[[package]] +category = "dev" +description = "sphinxcontrib-devhelp is a sphinx extension which outputs Devhelp document." +name = "sphinxcontrib-devhelp" +optional = false +python-versions = ">=3.5" +version = "1.0.2" + +[package.extras] +lint = ["flake8", "mypy", "docutils-stubs"] +test = ["pytest"] + +[[package]] +category = "dev" +description = "sphinxcontrib-htmlhelp is a sphinx extension which renders HTML help files" +name = "sphinxcontrib-htmlhelp" +optional = false +python-versions = ">=3.5" +version = "1.0.3" + +[package.extras] +lint = ["flake8", "mypy", "docutils-stubs"] +test = ["pytest", "html5lib"] + +[[package]] +category = "dev" +description = "A sphinx extension which renders display math in HTML via JavaScript" +name = "sphinxcontrib-jsmath" +optional = false +python-versions = ">=3.5" +version = "1.0.1" + +[package.extras] +test = ["pytest", "flake8", "mypy"] + +[[package]] +category = "dev" +description = "sphinxcontrib-qthelp is a sphinx extension which outputs QtHelp document." +name = "sphinxcontrib-qthelp" +optional = false +python-versions = ">=3.5" +version = "1.0.3" + +[package.extras] +lint = ["flake8", "mypy", "docutils-stubs"] +test = ["pytest"] + +[[package]] +category = "dev" +description = "sphinxcontrib-serializinghtml is a sphinx extension which outputs \"serialized\" HTML files (json and pickle)." +name = "sphinxcontrib-serializinghtml" +optional = false +python-versions = ">=3.5" +version = "1.1.4" + +[package.extras] +lint = ["flake8", "mypy", "docutils-stubs"] +test = ["pytest"] + [[package]] category = "dev" description = "Manage dynamic plugins for Python applications" @@ -830,6 +1047,19 @@ optional = false python-versions = "*" version = "3.7.4.2" +[[package]] +category = "dev" +description = "HTTP library with thread-safe connection pooling, file post, and more." +name = "urllib3" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, <4" +version = "1.25.10" + +[package.extras] +brotli = ["brotlipy (>=0.6.0)"] +secure = ["certifi", "cryptography (>=1.3.4)", "idna (>=2.0.0)", "pyOpenSSL (>=0.14)", "ipaddress"] +socks = ["PySocks (>=1.5.6,<1.5.7 || >1.5.7,<2.0)"] + [[package]] category = "dev" description = "Virtual Python Environment builder" @@ -910,11 +1140,15 @@ optional = ["pygments", "colorama"] tests = ["pytest", "pytest-cov", "codecov", "scikit-build", "cmake", "ninja", "pybind11"] [metadata] -content-hash = "79e46e67260312ba1b227f7717d0ba781a6ad5df352f83fe6e1b72a98ad1ec5b" +content-hash = "beb356287b912e1ce67c5997bd4d683dbc424ee24660a93d8f1fb4e511972762" lock-version = "1.0" python-versions = "^3.8" [metadata.files] +alabaster = [ + {file = "alabaster-0.7.12-py2.py3-none-any.whl", hash = "sha256:446438bdcca0e05bd45ea2de1668c1d9b032e1a9154c2c259092d77031ddd359"}, + {file = "alabaster-0.7.12.tar.gz", hash = "sha256:a661d72d58e6ea8a57f7a86e37d86716863ee5e92788398526d58b26a4e4dc02"}, +] appdirs = [ {file = "appdirs-1.4.4-py2.py3-none-any.whl", hash = "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128"}, {file = "appdirs-1.4.4.tar.gz", hash = "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41"}, @@ -946,6 +1180,10 @@ attrs = [ autoflake = [ {file = "autoflake-1.3.1.tar.gz", hash = "sha256:680cb9dade101ed647488238ccb8b8bfb4369b53d58ba2c8cdf7d5d54e01f95b"}, ] +babel = [ + {file = "Babel-2.8.0-py2.py3-none-any.whl", hash = "sha256:d670ea0b10f8b723672d3a6abeb87b565b244da220d76b4dba1b66269ec152d4"}, + {file = "Babel-2.8.0.tar.gz", hash = "sha256:1aac2ae2d0d8ea368fa90906567f5c08463d98ade155c0c4bfedd6a0f7160e38"}, +] bandit = [ {file = "bandit-1.6.2-py2.py3-none-any.whl", hash = "sha256:336620e220cf2d3115877685e264477ff9d9abaeb0afe3dc7264f55fa17a3952"}, {file = "bandit-1.6.2.tar.gz", hash = "sha256:41e75315853507aa145d62a78a2a6c5e3240fe14ee7c601459d0df9418196065"}, @@ -954,10 +1192,18 @@ black = [ {file = "black-19.10b0-py36-none-any.whl", hash = "sha256:1b30e59be925fafc1ee4565e5e08abef6b03fe455102883820fe5ee2e4734e0b"}, {file = "black-19.10b0.tar.gz", hash = "sha256:c2edb73a08e9e0e6f65a0e6af18b059b8b1cdd5bef997d7a0b181df93dc81539"}, ] +certifi = [ + {file = "certifi-2020.6.20-py2.py3-none-any.whl", hash = "sha256:8fc0819f1f30ba15bdb34cceffb9ef04d99f420f68eb75d901e9560b8749fc41"}, + {file = "certifi-2020.6.20.tar.gz", hash = "sha256:5930595817496dd21bb8dc35dad090f1c2cd0adfaf21204bf6732ca5d8ee34d3"}, +] cfgv = [ {file = "cfgv-3.2.0-py2.py3-none-any.whl", hash = "sha256:32e43d604bbe7896fe7c248a9c2276447dbef840feb28fe20494f62af110211d"}, {file = "cfgv-3.2.0.tar.gz", hash = "sha256:cf22deb93d4bcf92f345a5c3cd39d3d41d6340adc60c78bbbd6588c384fda6a1"}, ] +chardet = [ + {file = "chardet-3.0.4-py2.py3-none-any.whl", hash = "sha256:fc323ffcaeaed0e0a02bf4d117757b98aed530d9ed4531e3e15460124c106691"}, + {file = "chardet-3.0.4.tar.gz", hash = "sha256:84ab92ed1c4d4f16916e05906b6b75a6c0fb5db821cc65e70cbd64a3e2a5eaae"}, +] click = [ {file = "click-7.1.2-py2.py3-none-any.whl", hash = "sha256:dacca89f4bfadd5de3d7489b7c8a566eee0d3676333fbb50030263894c38c0dc"}, {file = "click-7.1.2.tar.gz", hash = "sha256:d2b5255c7c6349bc1bd1e59e08cd12acbbd63ce649f2588755783aa94dfb6b1a"}, @@ -1107,6 +1353,14 @@ identify = [ {file = "identify-1.4.25-py2.py3-none-any.whl", hash = "sha256:ccd88716b890ecbe10920659450a635d2d25de499b9a638525a48b48261d989b"}, {file = "identify-1.4.25.tar.gz", hash = "sha256:110ed090fec6bce1aabe3c72d9258a9de82207adeaa5a05cd75c635880312f9a"}, ] +idna = [ + {file = "idna-2.10-py2.py3-none-any.whl", hash = "sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0"}, + {file = "idna-2.10.tar.gz", hash = "sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6"}, +] +imagesize = [ + {file = "imagesize-1.2.0-py2.py3-none-any.whl", hash = "sha256:6965f19a6a2039c7d48bca7dba2473069ff854c36ae6f19d2cde309d998228a1"}, + {file = "imagesize-1.2.0.tar.gz", hash = "sha256:b1f6b5a4eab1f73479a50fb79fcf729514a900c341d8503d62a62dbc4127a2b1"}, +] iniconfig = [ {file = "iniconfig-1.0.1-py3-none-any.whl", hash = "sha256:80cf40c597eb564e86346103f609d74efce0f6b4d4f30ec8ce9e2c26411ba437"}, {file = "iniconfig-1.0.1.tar.gz", hash = "sha256:e5f92f89355a67de0595932a6c6c02ab4afddc6fcdc0bfc5becd0d60884d3f69"}, @@ -1115,6 +1369,10 @@ isort = [ {file = "isort-4.3.21-py2.py3-none-any.whl", hash = "sha256:6e811fcb295968434526407adb8796944f1988c5b65e8139058f2014cbe100fd"}, {file = "isort-4.3.21.tar.gz", hash = "sha256:54da7e92468955c4fceacd0c86bd0ec997b0e1ee80d97f67c35a78b719dccab1"}, ] +jinja2 = [ + {file = "Jinja2-2.11.2-py2.py3-none-any.whl", hash = "sha256:f0a4641d3cf955324a89c04f3d94663aa4d638abe8f733ecd3582848e1c37035"}, + {file = "Jinja2-2.11.2.tar.gz", hash = "sha256:89aab215427ef59c34ad58735269eb58b1a5808103067f7bb9d5836c651b3bb0"}, +] lazy-object-proxy = [ {file = "lazy-object-proxy-1.4.3.tar.gz", hash = "sha256:f3900e8a5de27447acbf900b4750b0ddfd7ec1ea7fbaf11dfa911141bc522af0"}, {file = "lazy_object_proxy-1.4.3-cp27-cp27m-macosx_10_13_x86_64.whl", hash = "sha256:a2238e9d1bb71a56cd710611a1614d1194dc10a175c1e08d75e1a7bcc250d442"}, @@ -1138,6 +1396,41 @@ lazy-object-proxy = [ {file = "lazy_object_proxy-1.4.3-cp38-cp38-win32.whl", hash = "sha256:5541cada25cd173702dbd99f8e22434105456314462326f06dba3e180f203dfd"}, {file = "lazy_object_proxy-1.4.3-cp38-cp38-win_amd64.whl", hash = "sha256:59f79fef100b09564bc2df42ea2d8d21a64fdcda64979c0fa3db7bdaabaf6239"}, ] +markupsafe = [ + {file = "MarkupSafe-1.1.1-cp27-cp27m-macosx_10_6_intel.whl", hash = "sha256:09027a7803a62ca78792ad89403b1b7a73a01c8cb65909cd876f7fcebd79b161"}, + {file = "MarkupSafe-1.1.1-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:e249096428b3ae81b08327a63a485ad0878de3fb939049038579ac0ef61e17e7"}, + {file = "MarkupSafe-1.1.1-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:500d4957e52ddc3351cabf489e79c91c17f6e0899158447047588650b5e69183"}, + {file = "MarkupSafe-1.1.1-cp27-cp27m-win32.whl", hash = "sha256:b2051432115498d3562c084a49bba65d97cf251f5a331c64a12ee7e04dacc51b"}, + {file = "MarkupSafe-1.1.1-cp27-cp27m-win_amd64.whl", hash = "sha256:98c7086708b163d425c67c7a91bad6e466bb99d797aa64f965e9d25c12111a5e"}, + {file = "MarkupSafe-1.1.1-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:cd5df75523866410809ca100dc9681e301e3c27567cf498077e8551b6d20e42f"}, + {file = "MarkupSafe-1.1.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:43a55c2930bbc139570ac2452adf3d70cdbb3cfe5912c71cdce1c2c6bbd9c5d1"}, + {file = "MarkupSafe-1.1.1-cp34-cp34m-macosx_10_6_intel.whl", hash = "sha256:1027c282dad077d0bae18be6794e6b6b8c91d58ed8a8d89a89d59693b9131db5"}, + {file = "MarkupSafe-1.1.1-cp34-cp34m-manylinux1_i686.whl", hash = "sha256:62fe6c95e3ec8a7fad637b7f3d372c15ec1caa01ab47926cfdf7a75b40e0eac1"}, + {file = "MarkupSafe-1.1.1-cp34-cp34m-manylinux1_x86_64.whl", hash = "sha256:88e5fcfb52ee7b911e8bb6d6aa2fd21fbecc674eadd44118a9cc3863f938e735"}, + {file = "MarkupSafe-1.1.1-cp34-cp34m-win32.whl", hash = "sha256:ade5e387d2ad0d7ebf59146cc00c8044acbd863725f887353a10df825fc8ae21"}, + {file = "MarkupSafe-1.1.1-cp34-cp34m-win_amd64.whl", hash = "sha256:09c4b7f37d6c648cb13f9230d847adf22f8171b1ccc4d5682398e77f40309235"}, + {file = "MarkupSafe-1.1.1-cp35-cp35m-macosx_10_6_intel.whl", hash = "sha256:79855e1c5b8da654cf486b830bd42c06e8780cea587384cf6545b7d9ac013a0b"}, + {file = "MarkupSafe-1.1.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:c8716a48d94b06bb3b2524c2b77e055fb313aeb4ea620c8dd03a105574ba704f"}, + {file = "MarkupSafe-1.1.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:7c1699dfe0cf8ff607dbdcc1e9b9af1755371f92a68f706051cc8c37d447c905"}, + {file = "MarkupSafe-1.1.1-cp35-cp35m-win32.whl", hash = "sha256:6dd73240d2af64df90aa7c4e7481e23825ea70af4b4922f8ede5b9e35f78a3b1"}, + {file = "MarkupSafe-1.1.1-cp35-cp35m-win_amd64.whl", hash = "sha256:9add70b36c5666a2ed02b43b335fe19002ee5235efd4b8a89bfcf9005bebac0d"}, + {file = "MarkupSafe-1.1.1-cp36-cp36m-macosx_10_6_intel.whl", hash = "sha256:24982cc2533820871eba85ba648cd53d8623687ff11cbb805be4ff7b4c971aff"}, + {file = "MarkupSafe-1.1.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:00bc623926325b26bb9605ae9eae8a215691f33cae5df11ca5424f06f2d1f473"}, + {file = "MarkupSafe-1.1.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:717ba8fe3ae9cc0006d7c451f0bb265ee07739daf76355d06366154ee68d221e"}, + {file = "MarkupSafe-1.1.1-cp36-cp36m-win32.whl", hash = "sha256:535f6fc4d397c1563d08b88e485c3496cf5784e927af890fb3c3aac7f933ec66"}, + {file = "MarkupSafe-1.1.1-cp36-cp36m-win_amd64.whl", hash = "sha256:b1282f8c00509d99fef04d8ba936b156d419be841854fe901d8ae224c59f0be5"}, + {file = "MarkupSafe-1.1.1-cp37-cp37m-macosx_10_6_intel.whl", hash = "sha256:8defac2f2ccd6805ebf65f5eeb132adcf2ab57aa11fdf4c0dd5169a004710e7d"}, + {file = "MarkupSafe-1.1.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:46c99d2de99945ec5cb54f23c8cd5689f6d7177305ebff350a58ce5f8de1669e"}, + {file = "MarkupSafe-1.1.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:ba59edeaa2fc6114428f1637ffff42da1e311e29382d81b339c1817d37ec93c6"}, + {file = "MarkupSafe-1.1.1-cp37-cp37m-win32.whl", hash = "sha256:b00c1de48212e4cc9603895652c5c410df699856a2853135b3967591e4beebc2"}, + {file = "MarkupSafe-1.1.1-cp37-cp37m-win_amd64.whl", hash = "sha256:9bf40443012702a1d2070043cb6291650a0841ece432556f784f004937f0f32c"}, + {file = "MarkupSafe-1.1.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6788b695d50a51edb699cb55e35487e430fa21f1ed838122d722e0ff0ac5ba15"}, + {file = "MarkupSafe-1.1.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:cdb132fc825c38e1aeec2c8aa9338310d29d337bebbd7baa06889d09a60a1fa2"}, + {file = "MarkupSafe-1.1.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:13d3144e1e340870b25e7b10b98d779608c02016d5184cfb9927a9f10c689f42"}, + {file = "MarkupSafe-1.1.1-cp38-cp38-win32.whl", hash = "sha256:596510de112c685489095da617b5bcbbac7dd6384aeebeda4df6025d0256a81b"}, + {file = "MarkupSafe-1.1.1-cp38-cp38-win_amd64.whl", hash = "sha256:e8313f01ba26fbbe36c7be1966a7b7424942f670f38e666995b88d012765b9be"}, + {file = "MarkupSafe-1.1.1.tar.gz", hash = "sha256:29872e92839765e546828bb7754a68c418d927cd064fd4708fab9fe9c8bb116b"}, +] mccabe = [ {file = "mccabe-0.6.1-py2.py3-none-any.whl", hash = "sha256:ab8a6258860da4b6677da4bd2fe5dc2c659cff31b3ee4f7f5d64e79735b80d42"}, {file = "mccabe-0.6.1.tar.gz", hash = "sha256:dd8d182285a0fe56bace7f45b5e7d1a6ebcbf524e8f3bd87eb0f125271b8831f"}, @@ -1233,6 +1526,10 @@ pytest-cov = [ {file = "pytest-cov-2.10.0.tar.gz", hash = "sha256:1a629dc9f48e53512fcbfda6b07de490c374b0c83c55ff7a1720b3fccff0ac87"}, {file = "pytest_cov-2.10.0-py2.py3-none-any.whl", hash = "sha256:6e6d18092dce6fad667cd7020deed816f858ad3b49d5b5e2b1cc1c97a4dba65c"}, ] +pytz = [ + {file = "pytz-2020.1-py2.py3-none-any.whl", hash = "sha256:a494d53b6d39c3c6e44c3bec237336e14305e4f29bbf800b599253057fbb79ed"}, + {file = "pytz-2020.1.tar.gz", hash = "sha256:c35965d010ce31b23eeb663ed3cc8c906275d6be1a34393a1d73a41febf4a048"}, +] pyyaml = [ {file = "PyYAML-5.3.1-cp27-cp27m-win32.whl", hash = "sha256:74809a57b329d6cc0fdccee6318f44b9b8649961fa73144a98735b0aaf029f1f"}, {file = "PyYAML-5.3.1-cp27-cp27m-win_amd64.whl", hash = "sha256:240097ff019d7c70a4922b6869d8a86407758333f02203e0fc6ff79c5dcede76"}, @@ -1269,6 +1566,10 @@ regex = [ {file = "regex-2020.7.14-cp38-cp38-win_amd64.whl", hash = "sha256:7a2dd66d2d4df34fa82c9dc85657c5e019b87932019947faece7983f2089a840"}, {file = "regex-2020.7.14.tar.gz", hash = "sha256:3a3af27a8d23143c49a3420efe5b3f8cf1a48c6fc8bc6856b03f638abc1833bb"}, ] +requests = [ + {file = "requests-2.24.0-py2.py3-none-any.whl", hash = "sha256:fe75cc94a9443b9246fc7049224f75604b113c36acb93f87b80ed42c44cbb898"}, + {file = "requests-2.24.0.tar.gz", hash = "sha256:b3559a131db72c33ee969480840fff4bb6dd111de7dd27c8ee1f820f4f00231b"}, +] restructuredtext-lint = [ {file = "restructuredtext_lint-1.3.1.tar.gz", hash = "sha256:470e53b64817211a42805c3a104d2216f6f5834b22fe7adb637d1de4d6501fb8"}, ] @@ -1284,6 +1585,38 @@ snowballstemmer = [ {file = "snowballstemmer-2.0.0-py2.py3-none-any.whl", hash = "sha256:209f257d7533fdb3cb73bdbd24f436239ca3b2fa67d56f6ff88e86be08cc5ef0"}, {file = "snowballstemmer-2.0.0.tar.gz", hash = "sha256:df3bac3df4c2c01363f3dd2cfa78cce2840a79b9f1c2d2de9ce8d31683992f52"}, ] +sphinx = [ + {file = "Sphinx-3.1.2-py3-none-any.whl", hash = "sha256:97dbf2e31fc5684bb805104b8ad34434ed70e6c588f6896991b2fdfd2bef8c00"}, + {file = "Sphinx-3.1.2.tar.gz", hash = "sha256:b9daeb9b39aa1ffefc2809b43604109825300300b987a24f45976c001ba1a8fd"}, +] +sphinx-autodoc-typehints = [ + {file = "sphinx-autodoc-typehints-1.11.0.tar.gz", hash = "sha256:bbf0b203f1019b0f9843ee8eef0cff856dc04b341f6dbe1113e37f2ebf243e11"}, + {file = "sphinx_autodoc_typehints-1.11.0-py3-none-any.whl", hash = "sha256:89e19370a55db4aef1be2094d8fb1fb500ca455c55b3fcc8d2600ff805227e04"}, +] +sphinxcontrib-applehelp = [ + {file = "sphinxcontrib-applehelp-1.0.2.tar.gz", hash = "sha256:a072735ec80e7675e3f432fcae8610ecf509c5f1869d17e2eecff44389cdbc58"}, + {file = "sphinxcontrib_applehelp-1.0.2-py2.py3-none-any.whl", hash = "sha256:806111e5e962be97c29ec4c1e7fe277bfd19e9652fb1a4392105b43e01af885a"}, +] +sphinxcontrib-devhelp = [ + {file = "sphinxcontrib-devhelp-1.0.2.tar.gz", hash = "sha256:ff7f1afa7b9642e7060379360a67e9c41e8f3121f2ce9164266f61b9f4b338e4"}, + {file = "sphinxcontrib_devhelp-1.0.2-py2.py3-none-any.whl", hash = "sha256:8165223f9a335cc1af7ffe1ed31d2871f325254c0423bc0c4c7cd1c1e4734a2e"}, +] +sphinxcontrib-htmlhelp = [ + {file = "sphinxcontrib-htmlhelp-1.0.3.tar.gz", hash = "sha256:e8f5bb7e31b2dbb25b9cc435c8ab7a79787ebf7f906155729338f3156d93659b"}, + {file = "sphinxcontrib_htmlhelp-1.0.3-py2.py3-none-any.whl", hash = "sha256:3c0bc24a2c41e340ac37c85ced6dafc879ab485c095b1d65d2461ac2f7cca86f"}, +] +sphinxcontrib-jsmath = [ + {file = "sphinxcontrib-jsmath-1.0.1.tar.gz", hash = "sha256:a9925e4a4587247ed2191a22df5f6970656cb8ca2bd6284309578f2153e0c4b8"}, + {file = "sphinxcontrib_jsmath-1.0.1-py2.py3-none-any.whl", hash = "sha256:2ec2eaebfb78f3f2078e73666b1415417a116cc848b72e5172e596c871103178"}, +] +sphinxcontrib-qthelp = [ + {file = "sphinxcontrib-qthelp-1.0.3.tar.gz", hash = "sha256:4c33767ee058b70dba89a6fc5c1892c0d57a54be67ddd3e7875a18d14cba5a72"}, + {file = "sphinxcontrib_qthelp-1.0.3-py2.py3-none-any.whl", hash = "sha256:bd9fc24bcb748a8d51fd4ecaade681350aa63009a347a8c14e637895444dfab6"}, +] +sphinxcontrib-serializinghtml = [ + {file = "sphinxcontrib-serializinghtml-1.1.4.tar.gz", hash = "sha256:eaa0eccc86e982a9b939b2b82d12cc5d013385ba5eadcc7e4fed23f4405f77bc"}, + {file = "sphinxcontrib_serializinghtml-1.1.4-py2.py3-none-any.whl", hash = "sha256:f242a81d423f59617a8e5cf16f5d4d74e28ee9a66f9e5b637a18082991db5a9a"}, +] stevedore = [ {file = "stevedore-3.2.0-py3-none-any.whl", hash = "sha256:c8f4f0ebbc394e52ddf49de8bcc3cf8ad2b4425ebac494106bbc5e3661ac7633"}, {file = "stevedore-3.2.0.tar.gz", hash = "sha256:38791aa5bed922b0a844513c5f9ed37774b68edc609e5ab8ab8d8fe0ce4315e5"}, @@ -1324,6 +1657,10 @@ typing-extensions = [ {file = "typing_extensions-3.7.4.2-py3-none-any.whl", hash = "sha256:6e95524d8a547a91e08f404ae485bbb71962de46967e1b71a0cb89af24e761c5"}, {file = "typing_extensions-3.7.4.2.tar.gz", hash = "sha256:79ee589a3caca649a9bfd2a8de4709837400dfa00b6cc81962a1e6a1815969ae"}, ] +urllib3 = [ + {file = "urllib3-1.25.10-py2.py3-none-any.whl", hash = "sha256:e7983572181f5e1522d9c98453462384ee92a0be7fac5f1413a1e35c56cc0461"}, + {file = "urllib3-1.25.10.tar.gz", hash = "sha256:91056c15fa70756691db97756772bb1eb9678fa585d9184f24534b100dc60f4a"}, +] virtualenv = [ {file = "virtualenv-20.0.30-py2.py3-none-any.whl", hash = "sha256:8cd7b2a4850b003a11be2fc213e206419efab41115cc14bca20e69654f2ac08e"}, {file = "virtualenv-20.0.30.tar.gz", hash = "sha256:7b54fd606a1b85f83de49ad8d80dbec08e983a2d2f96685045b262ebc7481ee5"}, diff --git a/pyproject.toml b/pyproject.toml index 200a42a..5c7d727 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -55,5 +55,9 @@ pytest = "^6.0.1" pytest-cov = "^2.10.0" xdoctest = { version="^0.13.0", extras=["optional"] } +# Documentation +sphinx = "^3.1.2" +sphinx-autodoc-typehints = "^1.11.0" + [tool.poetry.scripts] umd = "urban_meal_delivery.console:main" diff --git a/setup.cfg b/setup.cfg index f275247..3e1b9bf 100644 --- a/setup.cfg +++ b/setup.cfg @@ -93,8 +93,13 @@ extend-ignore = WPS306, # Putting logic into __init__.py files may be justified. WPS412, + # Allow multiple assignment, e.g., x = y = 123 + WPS429, per-file-ignores = + docs/conf.py: + # Allow shadowing built-ins and reading __*__ variables. + WPS125,WPS609, noxfile.py: # Type annotations are not strictly enforced. ANN0, ANN2, diff --git a/src/urban_meal_delivery/__init__.py b/src/urban_meal_delivery/__init__.py index 425f95f..d130f04 100644 --- a/src/urban_meal_delivery/__init__.py +++ b/src/urban_meal_delivery/__init__.py @@ -13,9 +13,11 @@ try: _pkg_info = _metadata.metadata(__name__) except _metadata.PackageNotFoundError: # pragma: no cover + __author__ = 'unknown' __pkg_name__ = 'unknown' __version__ = 'unknown' else: + __author__ = _pkg_info['author'] __pkg_name__ = _pkg_info['name'] __version__ = _pkg_info['version'] From 762fbf33b4e439d114e3488b973de615f68dbdfe Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Wed, 5 Aug 2020 01:45:01 +0200 Subject: [PATCH 11/13] Finalize the automated tasks in nox - only support Python 3.8 in the project as GitHub Actions has no 3.9 in its base image - re-order some sessions so that the order in `poetry run nox --list` is more intuitive - add extensive documentation to the noxfile.py module - add more whitespace to noxfile.py to make it easier to read - add generic maintainance tasks: + "init-project" => set up all pre-commit hooks + "clean-pwd" => ~ `git clean -X` with minor exceptions - upgrade isort to 5.3.0 --- noxfile.py | 222 ++++++++++++++++++++++++++++++++++++++--------------- 1 file changed, 162 insertions(+), 60 deletions(-) diff --git a/noxfile.py b/noxfile.py index 36f7813..96cb2f8 100644 --- a/noxfile.py +++ b/noxfile.py @@ -1,6 +1,58 @@ -"""Configure nox for the isolated test and lint environments.""" +"""Configure nox as the task runner, including CI and pre-commit hooks. + +Generic maintainance tasks: + +- "init-project": set up the pre-commit hooks + +- "clean-pwd": ~ `git clean -X` with minor exceptions + + +For local development, use the "format", "lint", and "test" sessions +as unified tasks to assure the quality of the source code: + +- "format" (autoflake, black, isort): + + + check all source files [default] + + accept extra arguments, e.g., `poetry run nox -s format -- noxfile.py`, + that are then interpreted as the paths the formatters and linters work + on recursively + +- "lint" (flake8, mypy, pylint): same as "format" + +- "test" (pytest, xdoctest): + + + run the entire test suite [default] + + accepts extra arguments, e.g., `poetry run nox -s test -- --no-cov`, + that are passed on to `pytest` and `xdoctest` with no changes + => may be paths or options + + +GitHub Actions implements a CI workflow: + +- "format", "lint", and "test" as above + +- "safety": check if dependencies contain known security vulnerabilites + +- "docs": build the documentation with sphinx + + +The pre-commit framework invokes the "pre-commit" and "pre-merge" sessions: + +- "pre-commit" before all commits: + + + triggers "format" and "lint" on staged source files + + => test coverage may be < 100% + +- "pre-merge" before all merges and pushes: + + + same as "pre-commit" + + plus: triggers "test", "safety", and "docs" (that ignore extra arguments) + + => test coverage is enforced to be 100% + +""" import contextlib +import glob import os import tempfile @@ -8,11 +60,6 @@ import nox from nox.sessions import Session -MAIN_PYTHON = '3.8' - -# Keep the project is forward compatible. -NEXT_PYTHON = '3.9' - PACKAGE_IMPORT_NAME = 'urban_meal_delivery' # Docs/sphinx locations. @@ -33,6 +80,8 @@ SRC_LOCATIONS = ( PYTEST_LOCATION, ) +PYTHON = '3.8' + # Use a unified .cache/ folder for all tools. nox.options.envdir = '.cache/nox' @@ -44,14 +93,13 @@ nox.options.error_on_external_run = True nox.options.sessions = ( 'format', 'lint', - f'test-{MAIN_PYTHON}', - f'test-{NEXT_PYTHON}', + 'test', 'safety', 'docs', ) -@nox.session(name='format', python=MAIN_PYTHON) +@nox.session(name='format', python=PYTHON) def format_(session): """Format source files with autoflake, black, and isort. @@ -59,11 +107,14 @@ def format_(session): Otherwise, they are interpreted as paths the formatters work on recursively. """ _begin(session) + # The formatting tools do not require the developed # package be installed in the virtual environment. _install_packages(session, 'autoflake', 'black', 'isort') + # Interpret extra arguments as locations of source files. locations = session.posargs or SRC_LOCATIONS + session.run('autoflake', '--version') session.run( 'autoflake', @@ -76,14 +127,16 @@ def format_(session): '--remove-unused-variables', *locations, ) + session.run('black', '--version') session.run('black', *locations) + with _isort_fix(session): # TODO (isort): Remove after upgrading session.run('isort', '--version') session.run('isort', *locations) -@nox.session(python=MAIN_PYTHON) +@nox.session(python=PYTHON) def lint(session): """Lint source files with flake8, mypy, and pylint. @@ -91,6 +144,7 @@ def lint(session): Otherwise, they are interpreted as paths the linters work on recursively. """ _begin(session) + # The linting tools do not require the developed # package be installed in the virtual environment. _install_packages( @@ -104,13 +158,17 @@ def lint(session): 'pylint', 'wemake-python-styleguide', ) + # Interpret extra arguments as locations of source files. locations = session.posargs or SRC_LOCATIONS + session.run('flake8', '--version') session.run('flake8', '--ignore=I0', *locations) # TODO (isort): Remove flag + with _isort_fix(session): # TODO (isort): Remove after upgrading session.run('isort', '--version') session.run('isort', '--check-only', *locations) + # For mypy, only lint *.py files to be packaged. mypy_locations = [ path for path in locations if path.startswith(PACKAGE_SOURCE_LOCATION) @@ -120,6 +178,7 @@ def lint(session): session.run('mypy', *mypy_locations) else: session.log('No paths to be checked with mypy') + # Ignore errors where pylint cannot import a third-party package due its # being run in an isolated environment. For the same reason, pylint is # also not able to determine the correct order of imports. @@ -133,7 +192,24 @@ def lint(session): ) -@nox.session(python=[MAIN_PYTHON, NEXT_PYTHON]) +@nox.session(name='pre-commit', python=PYTHON, venv_backend='none') +def pre_commit(session): + """Run the format and lint sessions. + + Source files must be well-formed before they enter git. + + Intended to be run as a pre-commit hook. + + Passed in extra arguments are forwarded. So, if it is run as a pre-commit + hook, only the currently staged source files are formatted and linted. + """ + # "format" and "lint" are run in sessions on their own as + # session.notify() creates new Session objects. + session.notify('format') + session.notify('lint') + + +@nox.session(python=PYTHON) def test(session): """Test the code base. @@ -151,21 +227,22 @@ def test(session): # `poetry install --no-dev` removes previously installed packages. # We keep things simple and forbid such usage. if session.virtualenv.reuse_existing: - raise RuntimeError( - 'The "test" and "pre-merge" sessions must be run without the "-r" option', - ) + raise RuntimeError('The "test" session must be run without the "-r" option') _begin(session) + # The testing tools require the developed package and its # non-develop dependencies be installed in the virtual environment. session.run('poetry', 'install', '--no-dev', external=True) _install_packages( session, 'packaging', 'pytest', 'pytest-cov', 'xdoctest[optional]', ) + # Interpret extra arguments as options for pytest. # They are "dropped" by the hack in the pre_merge() function # if this function is run within the "pre-merge" session. posargs = () if session.env.get('_drop_posargs') else session.posargs + args = posargs or ( '--cov', '--no-cov-on-fail', @@ -176,60 +253,22 @@ def test(session): ) session.run('pytest', '--version') session.run('pytest', *args) + # For xdoctest, the default arguments are different from pytest. args = posargs or [PACKAGE_IMPORT_NAME] session.run('xdoctest', '--version') session.run('xdoctest', '--quiet', *args) # --quiet => less verbose output -@nox.session(name='pre-commit', python=MAIN_PYTHON, venv_backend='none') -def pre_commit(session): - """Source files must be well-formed before they enter git. - - Intended to be run as a pre-commit hook. - - This session is a wrapper that triggers the "format" and "lint" sessions. - - Passed in extra arguments are forwarded. So, if it is run as a pre-commit - hook, only the currently staged source files are formatted and linted. - """ - # "format" and "lint" are run in sessions on their own as - # session.notify() creates new Session objects. - session.notify('format') - session.notify('lint') - - -@nox.session(name='pre-merge', python=MAIN_PYTHON) -def pre_merge(session): - """The test suite must pass before merges are made. - - Intended to be run either as a pre-merge or pre-push hook. - - First, this session triggers the "format" and "lint" sessions via - the "pre-commit" session. - - Then, it runs the "test" session ignoring any extra arguments passed in - so that the entire test suite is executed. - """ - session.notify('pre-commit') - # Little hack to not work with the extra arguments provided - # by the pre-commit framework. Create a flag in the - # env(ironment) that must contain only `str`-like objects. - session.env['_drop_posargs'] = 'true' - # Cannot use session.notify() to trigger the "test" session - # as that would create a new Session object without the flag - # in the env(ironment). Instead, run the test() function within - # the "pre-merge" session. - test(session) - - -@nox.session(python=MAIN_PYTHON) +@nox.session(python=PYTHON) def safety(session): """Check the dependencies for known security vulnerabilities.""" _begin(session) + # We do not pin the version of `safety` to always check with # the latest version. The risk this breaks the CI is rather low. session.install('safety') + with tempfile.NamedTemporaryFile() as requirements_txt: session.run( 'poetry', @@ -244,7 +283,7 @@ def safety(session): ) -@nox.session(python=MAIN_PYTHON) +@nox.session(python=PYTHON) def docs(session): """Build the documentation with sphinx.""" # The latest version of the package needs to be installed @@ -267,6 +306,69 @@ def docs(session): print(f'Docs are available at {os.getcwd()}/{DOCS_BUILD}index.html') # noqa:WPS421 +@nox.session(name='pre-merge', python=PYTHON) +def pre_merge(session): + """Run the format, lint, test, safety, and docs sessions. + + Intended to be run either as a pre-merge or pre-push hook. + + Ignores the paths passed in by the pre-commit framework + for the test, safety, and docs sessions so that the + entire test suite is executed. + """ + # Re-using an old environment is not so easy here as the "test" session + # runs `poetry install --no-dev`, which removes previously installed packages. + if session.virtualenv.reuse_existing: + raise RuntimeError( + 'The "pre-merge" session must be run without the "-r" option', + ) + + session.notify('format') + session.notify('lint') + session.notify('safety') + session.notify('docs') + + # Little hack to not work with the extra arguments provided + # by the pre-commit framework. Create a flag in the + # env(ironment) that must contain only `str`-like objects. + session.env['_drop_posargs'] = 'true' + + # Cannot use session.notify() to trigger the "test" session + # as that would create a new Session object without the flag + # in the env(ironment). Instead, run the test() function within + # the "pre-merge" session. + test(session) + + +@nox.session(name='init-project', python=PYTHON, venv_backend='none') +def init_project(session): + """Install the pre-commit hooks.""" + for type_ in ('pre-commit', 'pre-merge-commit', 'pre-push'): + session.run('poetry', 'run', 'pre-commit', 'install', f'--hook-type={type_}') + + +@nox.session(name='clean-pwd', python=PYTHON, venv_backend='none') +def clean_pwd(session): + """Remove (almost) all glob patterns listed in .gitignore. + + The difference compared to `git clean -X` is that this task + does not remove pyenv's .python-version file and poetry's + virtual environment. + """ + exclude = frozenset(('.python-version', '.venv', 'venv')) + + with open('.gitignore') as file_handle: + paths = file_handle.readlines() + + for path in paths: + path = path.strip() + if path.startswith('#') or path in exclude: + continue + + for expanded in glob.glob(path): + session.run(f'rm -rf {expanded}') + + def _begin(session): """Show generic info about a session.""" if session.posargs: @@ -326,11 +428,11 @@ def _install_packages(session: Session, *packages_or_pip_args: str, **kwargs) -> # TODO (isort): Remove this fix after -# upgrading to isort ^5.2.2 in pyproject.toml. +# upgrading to isort ^5.3.0 in pyproject.toml. @contextlib.contextmanager def _isort_fix(session): - """Temporarily upgrade to isort 5.2.2.""" - session.install('isort==5.2.2') + """Temporarily upgrade to isort 5.3.0.""" + session.install('isort==5.3.0') try: yield finally: From 44f32cecd76944da6a5598ca7cb8bd939e234940 Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Wed, 5 Aug 2020 15:38:28 +0200 Subject: [PATCH 12/13] Enable CI with GitHub Actions --- .github/workflows/tests.yml | 14 ++++++++++++++ 1 file changed, 14 insertions(+) create mode 100644 .github/workflows/tests.yml diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml new file mode 100644 index 0000000..0724c09 --- /dev/null +++ b/.github/workflows/tests.yml @@ -0,0 +1,14 @@ +name: CI +on: push +jobs: + tests: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + - uses: actions/setup-python@v1 + with: + python-version: 3.8 + architecture: x64 + - run: pip install nox==2020.5.24 + - run: pip install poetry==1.0.10 + - run: nox From 4cf0cf4b08a06f0a1b3235e9b35f4ccbac6841f6 Mon Sep 17 00:00:00 2001 From: Alexander Hess Date: Wed, 5 Aug 2020 15:53:34 +0200 Subject: [PATCH 13/13] Finalize release 0.1.0 --- pyproject.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pyproject.toml b/pyproject.toml index 5c7d727..7981fc3 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -9,7 +9,7 @@ target-version = ["py38"] [tool.poetry] name = "urban-meal-delivery" -version = "0.1.0.dev0" +version = "0.1.0" authors = ["Alexander Hess "] description = "Optimizing an urban meal delivery platform"