-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathupkeep.py
208 lines (170 loc) · 6.4 KB
/
upkeep.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
import argparse
import datetime
import functools
import re
import sys
import traceback
import typing
import holidays
import github
class PennHolidays(holidays.UnitedStates):
def _populate(self, year):
super()._populate(year)
# See /~https://github.com/greenelab/scrum/issues/114
for day in range(26, 32):
self[datetime.date(year, 12, day)] = 'Special Winter Vacation'
holiday_names = {
'Independence Day',
'Labor Day',
'Thanksgiving',
'Christmas Day',
"New Year's Day",
'Martin Luther King, Jr. Day',
'Memorial Day',
'Special Winter Vacation',
}
penn_holidays = PennHolidays()
def get_today() -> datetime.date:
"""
Returns the datetime.date for today. Needed since tests cannot mock a
builtin type: http://stackoverflow.com/a/24005764/4651668
"""
return datetime.date.today()
def is_holiday(date: datetime.date) -> bool:
"""
Return True or False for whether a date is a holiday
"""
name = penn_holidays.get(date)
if not name:
return False
name = name.replace(' (Observed)', '')
return name in holiday_names
def is_workday(date) -> bool:
"""
Return boolean for whether a date is a workday.
"""
if date.weekday() in holidays.WEEKEND:
return False
if is_holiday(date):
return False
return True
@functools.lru_cache()
def issue_title_to_date(title: str) -> typing.Optional[datetime.date]:
"""
Return a datetime.date object from a Scrum issue title.
"""
pattern = re.compile(r'([0-9]{4})-([0-9]{2})-([0-9]{2}):')
match = pattern.match(title)
if not match:
return None
return datetime.date(*map(int, match.groups()))
def close_old_issues(issues, lifespan: int):
"""
Close scrum issues older than the number of days specified by lifespan.
"""
lifespan = datetime.timedelta(days=lifespan)
today = get_today()
for issue in issues:
if issue.state == 'closed':
continue
title = issue.title
date = issue_title_to_date(title)
if not date:
continue
if today - date > lifespan:
print('Closing', title, file=sys.stderr)
try:
issue.edit(state='closed')
except Exception:
print('Closing issue failed:\n{}'.format(traceback.format_exc()), file=sys.stderr)
def create_scrum_issue(
repo: github.Repository.Repository,
date: datetime.date,
previous_issue: github.Issue.Issue = None,
) -> typing.Optional[github.Issue.Issue]:
"""
Create a scrum issue for the given date.
If not None, previous_issue is used to set an issue body
that refers to the previous issue.
"""
kwargs = {'title': f"{date}: e-scrum for {date:%A, %B %-d, %Y}"}
if previous_issue:
kwargs['body'] = 'Preceeding e-scrum in {}.'.format(previous_issue.html_url)
print('Creating {title!r}'.format(**kwargs), file=sys.stderr)
try:
return repo.create_issue(**kwargs)
except Exception:
print('Creating issue failed:\n{}'.fomrat(traceback.format_exc()), file=sys.stderr)
def get_future_dates_without_issues(issues, workdays_ahead: int = 2):
"""
Look through issues and yield the dates of future workdays (includes today)
that don't have open issues.
"""
future_dates = set(get_upcoming_workdays(workdays_ahead))
future_dates -= {issue_title_to_date(x.title) for x in issues}
return sorted(future_dates)
def get_upcoming_workdays(workdays_ahead: int = 2) -> typing.Iterator[datetime.date]:
"""
Return a generator of the next number of workdays specified by
workdays_ahead. The current day is yielded first, if a workday,
and does not count as one of workdays_ahead.
"""
date = get_today()
if is_workday(date):
yield date
i = 0
while i < workdays_ahead:
date += datetime.timedelta(days=1)
if is_workday(date):
yield date
i += 1
if __name__ == '__main__':
parser = argparse.ArgumentParser()
parser.add_argument('--username', default='scrum-lord')
parser.add_argument(
'--token', help='GitHub personal access token for --username')
parser.add_argument('--repository', default='greenelab/scrum')
parser.add_argument('--lifespan', type=int, default=7)
parser.add_argument('--workdays-ahead', type=int, default=2)
parser.add_argument('--upkeep-file', type=str, default='uptime.txt')
args = parser.parse_args()
gh = github.Github(args.username, args.token)
user = gh.get_user()
# Get greenelab/scrum repository. Could not find a better way
repo, = [
repo for repo in user.get_repos()
if repo.full_name == args.repository
]
# Get open issues
open_issues = list(repo.get_issues(state='open'))
# Close old issues
close_old_issues(open_issues, args.lifespan)
# Get n most recent issues (open or closed), where n = 10 + --workdays-ahead
# to help ensure the most recent existing e-scrum issue is included even when other
# non e-scrum issues exist
issues = repo.get_issues(state='all', sort='number', direction='desc')
issues = issues[:min(10 + args.workdays_ahead, issues.totalCount)]
date_issue_pairs = [(issue_title_to_date(issue.title), issue) for issue in issues]
# Filter issues that are not scrum entries
filtered_date_issue_pairs = [(date, issue) for date, issue in date_issue_pairs if date]
# Issue objects are not comparable, so we need to sort by date only
date_issue_pairs = sorted(filtered_date_issue_pairs, key=lambda x: x[0])
# Detect previous issue for creation of the first upcoming issue
previous_issue = None
if date_issue_pairs:
_, previous_issue = date_issue_pairs[-1]
# Create upcoming issues
dates = get_future_dates_without_issues(issues, args.workdays_ahead)
for date in dates:
previous_issue = create_scrum_issue(repo, date, previous_issue)
# Create a small, meaningless change to keep Github Actions from disabling
# the repo for inactivity
with open(args.upkeep_file) as in_file:
message = in_file.readline().strip()
days = int(message.split(' ')[3])
days += 1
new_message = "It has been "
new_message += str(days)
new_message += " days since I last had to tinker with the scrum bot.\n"
with open(args.upkeep_file, 'w') as out_file:
out_file.write(new_message)