-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathtasks.py
238 lines (188 loc) · 8.08 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
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
import os
from colorama import Fore, just_fix_windows_console
from invoke import task
# from pathlib import Path
def execute_bump_hack(c):
"""A little hack that combines commitizen-tools and standard-version
commitizen-tools understands Python stuff, but I don't like the
generated changelogs. I had no time to look at how to cusstomize them, so I
decided to use standard-version (from the Javascript world). Unfortunately,
standard-version doesn't understand Python stuff, and since I didn't have
time to write my own updater for python files and toml files, I have to
make the two work together!
This requires standard-version to be installed globally on your system:
``npm i -g standard-version``
If you're setting it up for the first time on another project, you will probably
encounter problems generating the entire changelog. See how Łukasz Nojek came up
with a hack to deal with this:
https://lukasznojek.com/blog/2020/03/how-to-regenerate-changelog-using-standard-version/
The formula (workflow) for is as follows:
1. cz bump --files-only
2. git add pyproject.toml readme_coverage_badger/__init__.py readme_coverage_badger/__main__.py
3. standard-version --commit-all --release-as <result from cz if not none>
4. git push --follow-tags origin master
# TODO: add additional options here, which can passed to either cz or standard version
"""
print(f"{Fore.MAGENTA}Attempting to bump using commitizen-tools ...{Fore.RESET}")
c.run("cz bump --files-only > .bump_result.txt", pty=True)
str_of_interest = "increment detected: "
result = ""
with open(".bump_result.txt", "r") as br:
for line in br:
if str_of_interest in line:
result = line
break
release_type = result.replace(str_of_interest, "").strip("\n").lower()
print(f"cz bump result: {release_type}")
if release_type == "none":
print(f"{Fore.YELLOW}No increment detected, cannot bump{Fore.RESET}")
elif release_type in ["major", "minor", "patch"]:
print(f"{Fore.GREEN}Looks like the bump command worked!{Fore.RESET}")
print(f"{Fore.GREEN}Now handing over to standard-version ...{Fore.RESET}")
# first, stage the bumped files
c.run(
"git add pyproject.toml readme_coverage_badger/__init__.py readme_coverage_badger/__main__.py",
pty=True,
)
# now we can pass result to standard-release
print(f"{Fore.GREEN}let me retrieve the tag we're bumping from ...{Fore.RESET}")
# get_previous_tag = c.run(
# "git describe --abbrev=0 --tags `git rev-list --tags --skip=1 --max-count=1`",
# pty=True,
# )
get_current_tag = c.run(
"git describe --abbrev=0 --tags `git rev-list --tags --skip=0 --max-count=1`",
pty=True,
)
previous_tag = get_current_tag.stdout.rstrip()
c.run(
f'standard-version --commit-all --release-as {release_type} --releaseCommitMessageFormat "bump: ✈️ {previous_tag} → v{{{{currentTag}}}}"',
pty=True,
)
# push to origin
c.run("git push --follow-tags origin master", pty=True)
else:
print(
f"{Fore.RED}Something went horribly wrong, please figure it out yourself{Fore.RESET}"
)
print(f"{Fore.RED}Bump failed!{Fore.RESET}")
# clean up
c.run("rm -vf .bump_result.txt", pty=True)
@task(
help={
"version": "show program's version number and exit",
"plain": "Plain colour mode. Standard green badge",
"help": "show this help message and exit",
}
)
def run(c, version=False, plain=False, help=False):
"""run the program"""
if version:
c.run("python readme_coverage_badger/__main__.py -v", pty=True)
elif plain:
c.run("python readme_coverage_badger/__main__.py -p", pty=True)
elif help:
c.run("python readme_coverage_badger/__main__.py -h", pty=True)
else:
c.run("python readme_coverage_badger/__main__.py", pty=True)
@task
def test(c):
"""run tests"""
c.run("pytest", pty=True)
@task(help={"fix": "let black and isort format your files"})
def lint(c, fix=False):
"""flake8, black, isort and mypy"""
if fix:
c.run("black .", pty=True)
c.run("isort --profile black .", pty=True)
else:
c.run("mypy readme_coverage_badger", pty=True)
c.run("black . --check", pty=True)
c.run("isort --check-only --profile black .", pty=True)
c.run("flake8 tests readme_coverage_badger", pty=True)
# TODO: create a "clean" collection comprising the next three tasks below
@task
def clean_build(c):
"""remove build artifacts"""
# TODO: Make this more portable, as it'll probably break on a Windows prompt
c.run("rm -fr build/", pty=True)
c.run("rm -fr dist/", pty=True)
c.run("rm -fr .eggs/", pty=True)
c.run("find . -name '*.egg-info' -exec rm -fr {} +", pty=True)
c.run("find . -name '*.egg' -exec rm -f {} +", pty=True)
@task
def clean_pyc(c):
"""remove Python file artifacts"""
# TODO: Make this more portable, as it'll probably break on a Win32 prompt
c.run("find . -name '*.pyc' -exec rm -f {} +", pty=True)
c.run("find . -name '*.pyo' -exec rm -f {} +", pty=True)
c.run("find . -name '*~' -exec rm -f {} +", pty=True)
c.run("find . -name '__pycache__' -exec rm -fr {} +", pty=True)
@task
def clean_test(c):
"""remove test and coverage artifacts"""
# TODO: Make this more portable, as it'll probably break on a Win32 prompt
c.run("rm -fr .tox/", pty=True)
c.run("rm -f .coverage", pty=True)
c.run("rm -f coverage.xml", pty=True)
c.run("rm -fr htmlcov/", pty=True)
c.run("rm -fr .pytest_cache", pty=True)
@task
def bump(c):
"""Use Commitizen Tools & standard-version to bump version and generate changelog
Run this task when you want to prepare a release.
First we check that there are no unstaged files in your repo before running
"""
just_fix_windows_console()
# TODO: this can be part of the "release" collection, where we bump, build and release
unstaged_str = "not staged for commit"
uncommitted_str = "to be committed"
check = c.run("git status", pty=True)
if unstaged_str not in check.stdout or uncommitted_str not in check.stdout:
execute_bump_hack(c)
else:
print(
f"{Fore.RED}Sorry mate, please ensure there are no pending git operations{Fore.RESET}"
)
@task
def dist(c):
"""builds source and wheel package"""
# TODO: provide a "clean" option, and improve portability
c.run("python -m build", pty=True)
c.run("ls -lah dist", pty=True)
c.run("pwd", pty=True)
@task
def release(c):
"""(package and) upload an official release"""
# TODO: provide a "dist" option, so that we build first then release
c.run("twine upload dist/*", pty=True)
@task
def release_testpypi(c):
"""(package and) upload a test release on https://test.pypi.org/"""
# TODO: provide a "dist" option, so that we build first then release
c.run("twine upload -r testpypi dist/*", pty=True)
@task
def get_release_notes(c):
"""extract content from CHANGELOG.md for use in Github Releases
we read the file and loop through line by line
we wanna extract content beginning from the first Heading 2 text
to the last line before the next Heading 2 text
"""
pattern_to_match = "## [v"
count = 0
lines = []
heading_text = "## What's changed in this release\n"
lines.append(heading_text)
with open("CHANGELOG.md", "r") as c:
for line in c:
if pattern_to_match in line and count == 0:
count += 1
elif pattern_to_match not in line and count == 1:
lines.append(line)
elif pattern_to_match in line and count == 1:
break
# home = str(Path.home())
# release_notes = os.path.join(home, "LATEST_RELEASE_NOTES.md")
release_notes = os.path.join("../", "LATEST_RELEASE_NOTES.md")
with open(release_notes, "w") as f:
print("".join(lines), file=f, end="")