-
Notifications
You must be signed in to change notification settings - Fork 3
/
tasks.py
186 lines (148 loc) · 4.12 KB
/
tasks.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
"""
Tasks for maintaining the project.
Execute 'invoke --list' for guidance on using Invoke
"""
import shutil
import platform
from invoke import task
from pathlib import Path
import webbrowser
ROOT_DIR = Path(__file__).parent
SETUP_FILE = ROOT_DIR.joinpath("setup.py")
TEST_DIR = ROOT_DIR.joinpath("tests")
SOURCE_DIR = ROOT_DIR.joinpath("pydantic_panel")
TOX_DIR = ROOT_DIR.joinpath(".tox")
COVERAGE_FILE = ROOT_DIR.joinpath(".coverage")
COVERAGE_DIR = ROOT_DIR.joinpath("htmlcov")
COVERAGE_REPORT = COVERAGE_DIR.joinpath("index.html")
DOCS_DIR = ROOT_DIR.joinpath("docs")
DOCS_BUILD_DIR = DOCS_DIR.joinpath("_build")
DOCS_INDEX = DOCS_BUILD_DIR.joinpath("index.html")
PYTHON_DIRS = [str(d) for d in [SOURCE_DIR, TEST_DIR]]
def _delete_file(file):
try:
file.unlink(missing_ok=True)
except TypeError:
# missing_ok argument added in 3.8
try:
file.unlink()
except FileNotFoundError:
pass
def _run(c, command):
return c.run(command, pty=platform.system() != "Windows")
@task(help={"check": "Checks if source is formatted without applying changes"})
def format(c, check=False):
"""
Format code
"""
python_dirs_string = " ".join(PYTHON_DIRS)
# Run yapf
yapf_options = "--recursive {}".format("--diff" if check else "--in-place")
_run(c, "yapf {} {}".format(yapf_options, python_dirs_string))
# Run isort
isort_options = "--recursive {}".format("--check-only --diff" if check else "")
_run(c, "isort {} {}".format(isort_options, python_dirs_string))
@task
def lint_flake8(c):
"""
Lint code with flake8
"""
_run(c, "flake8 {}".format(" ".join(PYTHON_DIRS)))
@task
def lint_pylint(c):
"""
Lint code with pylint
"""
_run(c, "pylint {}".format(" ".join(PYTHON_DIRS)))
@task(lint_flake8, lint_pylint)
def lint(c):
"""
Run all linting
"""
@task
def test(c):
"""
Run tests
"""
_run(c, "pytest")
@task(help={"publish": "Publish the result via coveralls"})
def coverage(c, publish=False):
"""
Create coverage report
"""
_run(c, "coverage run --source {} -m pytest".format(SOURCE_DIR))
_run(c, "coverage report")
if publish:
# Publish the results via coveralls
_run(c, "coveralls")
else:
# Build a local report
_run(c, "coverage html")
webbrowser.open(COVERAGE_REPORT.as_uri())
@task(help={"launch": "Launch documentation in the web browser"})
def docs(c, launch=True):
"""
Generate documentation
"""
_run(c, "sphinx-build -b html {} {}".format(DOCS_DIR, DOCS_BUILD_DIR))
if launch:
webbrowser.open(DOCS_INDEX.as_uri())
@task
def clean_docs(c):
"""
Clean up files from documentation builds
"""
_run(c, "rm -fr {}".format(DOCS_BUILD_DIR))
@task
def clean_build(c):
"""
Clean up files from package building
"""
_run(c, "rm -fr build/")
_run(c, "rm -fr dist/")
_run(c, "rm -fr .eggs/")
_run(c, "find . -name '*.egg-info' -exec rm -fr {} +")
_run(c, "find . -name '*.egg' -exec rm -f {} +")
@task
def clean_python(c):
"""
Clean up python file artifacts
"""
_run(c, "find . -name '*.pyc' -exec rm -f {} +")
_run(c, "find . -name '*.pyo' -exec rm -f {} +")
_run(c, "find . -name '*~' -exec rm -f {} +")
_run(c, "find . -name '__pycache__' -exec rm -fr {} +")
@task
def clean_tests(c):
"""
Clean up files from testing
"""
_delete_file(COVERAGE_FILE)
shutil.rmtree(TOX_DIR, ignore_errors=True)
shutil.rmtree(COVERAGE_DIR, ignore_errors=True)
@task(pre=[clean_build, clean_python, clean_tests, clean_docs])
def clean(c):
"""
Runs all clean sub-tasks
"""
pass
@task(clean)
def dist(c):
"""
Build source and wheel packages
"""
_run(c, "poetry build")
@task(pre=[clean, dist])
def release(c):
"""
Make a release of the python package to pypi
"""
_run(c, "poetry publish")
@task(help={"version": "major/minor/path or explicit version"})
def bump(c, version="patch"):
"""
Bump version
"""
_run(c, "bump2version {}".format(version))
_run(c, "git push")
_run(c, "git push --tags")