Add ptl_nominations_last_day email template

This consumes (and modifies) render-statistics.  Factoring this out will
be done at the end of the series

Change-Id: I878de8288f1ac047d844085f05261a6d3b1daac1
This commit is contained in:
Tony Breeds 2019-02-08 15:44:41 +11:00
parent 069e2828f7
commit ffe1aa3c8b
2 changed files with 66 additions and 42 deletions

View File

@ -69,6 +69,53 @@ def collect_project_stats(basedir, verbose, projects):
', '.join(candidates))) ', '.join(candidates)))
def election_summary():
now = datetime.datetime.now(tz=pytz.utc)
now = now.replace(microsecond=0)
event = utils.get_event('PTL Nominations')
start, end = event['start'], event['end']
duration = (end - start)
remaining = (end - now)
progress = (duration - remaining)
p_progress = as_percentage(progress.total_seconds(),
duration.total_seconds())
p_candidate = as_percentage(counts['with_candidate'], counts['projects'])
p_nominations = as_percentage(counts['nominations'], counts['projects'])
need_election.sort()
without_candidate.sort()
output = ""
output += ("%-25s @ %s\n" % ("Nominations started", as_utctime(start)))
output += ("%-25s @ %s\n" % ("Nominations end", as_utctime(end)))
output += ("%-25s : %s\n" % ("Nominations duration", duration))
output += ("%-25s : %s\n" % ("Nominations remaining", remaining))
output += ("%-25s : %6.2f%%\n" % ("Nominations progress", p_progress))
output += ("-" * 51)
output += ("\n")
output += ("%-25s : %5d\n" % ("Projects[1]", counts['projects']))
output += ("%-25s : %5d (%6.2f%%)\n" % ("Projects with candidates",
counts['with_candidate'],
p_candidate))
output += ("%-25s : %5d (%6.2f%%)\n" % ("Projects with election",
counts['nominations'],
p_nominations))
output += ("-" * 51)
output += ("\n")
output += ("%-25s : %d (%s)\n" % ("Need election",
len(need_election),
" ".join(need_election)))
output += ("%-25s : %d (%s)\n" % ("Need appointment",
len(without_candidate),
" ".join(without_candidate)))
output += ("=" * 51)
output += ("\n")
output += ("%-25s @ %s\n" % ("Stats gathered", as_utctime(now)))
return output
def main(): def main():
parser = argparse.ArgumentParser(description='Investigate PTL Nominations') parser = argparse.ArgumentParser(description='Investigate PTL Nominations')
parser.add_argument('-v', '--verbose', action="count", default=0, parser.add_argument('-v', '--verbose', action="count", default=0,
@ -94,45 +141,9 @@ def main():
args.basedir = os.path.expanduser(args.basedir) args.basedir = os.path.expanduser(args.basedir)
collect_project_stats(args.basedir, args.verbose, args.projects) collect_project_stats(args.basedir, args.verbose, args.projects)
now = datetime.datetime.now(tz=pytz.utc)
now = now.replace(microsecond=0)
event = utils.get_event('PTL Nominations')
start, end = event['start'], event['end']
duration = (end - start)
remaining = (end - now)
progress = (duration - remaining)
p_progress = as_percentage(progress.total_seconds(),
duration.total_seconds())
p_candidate = as_percentage(counts['with_candidate'], counts['projects'])
p_nominations = as_percentage(counts['nominations'], counts['projects'])
need_election.sort()
without_candidate.sort()
if args.verbose: if args.verbose:
print("-" * 51) print("-" * 51)
print("%-25s @ %s" % ("Nominations started", as_utctime(start))) print(election_summary())
print("%-25s @ %s" % ("Nominations end", as_utctime(end)))
print("%-25s : %s" % ("Nominations duration", duration))
print("%-25s : %s" % ("Nominations remaining", remaining))
print("%-25s : %6.2f%%" % ("Nominations progress", p_progress))
print("-" * 51)
print("%-25s : %5d" % ("Projects[1]", counts['projects']))
print("%-25s : %5d (%6.2f%%)" % ("Projects with candidates",
counts['with_candidate'],
p_candidate))
print("%-25s : %5d (%6.2f%%)" % ("Projects with election",
counts['nominations'],
p_nominations))
print("-" * 51)
print("%-25s : %d (%s)" % ("Need election", len(need_election),
" ".join(need_election)))
print("%-25s : %d (%s)" % ("Need appointment", len(without_candidate),
" ".join(without_candidate)))
print("=" * 51)
print("%-25s @ %s" % ("Stats gathered", as_utctime(now)))
print("") print("")
print("[1] These numbers include the following projects that have a " print("[1] These numbers include the following projects that have a "
"candidate that is approved my only a single election official:" "candidate that is approved my only a single election official:"

View File

@ -2,8 +2,10 @@ from __future__ import print_function
from __future__ import unicode_literals from __future__ import unicode_literals
import argparse import argparse
import os
import sys import sys
from openstack_election.cmds import render_statistics as stats
from openstack_election import config from openstack_election import config
from openstack_election import utils from openstack_election import utils
@ -35,6 +37,11 @@ if conf['election_type'] == 'tc':
release=conf['release'], release=conf['release'],
) )
elif conf['election_type'] == 'ptl': elif conf['election_type'] == 'ptl':
# NOTE(tonyb): We need an empty item last to ensure the path ends in a
# tailing '/'
stats.collect_project_stats(os.path.join(utils.CANDIDATE_PATH,
conf['release'], ''),
False, [])
ptl_fmt_args = dict( ptl_fmt_args = dict(
nom_end_date=utils.get_event('PTL Nominations')['end_str'], nom_end_date=utils.get_event('PTL Nominations')['end_str'],
time_frame=time_frame, time_frame=time_frame,
@ -42,6 +49,9 @@ elif conf['election_type'] == 'ptl':
ending_release=end_release, ending_release=end_release,
future_release=end_release.lower(), future_release=end_release.lower(),
email_deadline=conf['timeframe']['email_deadline'], email_deadline=conf['timeframe']['email_deadline'],
num_projects_without_candidates=len(stats.without_candidate),
election_summary_stats=stats.election_summary(),
leaderless_url=LEADERLESS_URL,
) )
@ -114,7 +124,7 @@ Happy running,
print(email_text % (ptl_fmt_args)) print(email_text % (ptl_fmt_args))
def ptl_nominations_last_days(num_projects_without_candidates): def ptl_nominations_last_days():
email_text = """ email_text = """
A quick reminder that we are in the last hours for PTL candidate A quick reminder that we are in the last hours for PTL candidate
nominations. nominations.
@ -126,19 +136,21 @@ Make sure your nomination has been submitted to the openstack/election
repository and approved by election officials. repository and approved by election officials.
Election statistics[2]: Election statistics[2]:
%(election_summary_stats)s
This means that with approximately 2 days left, %s projects will This means that with approximately 2 days left, %(num_projects_without_candidates)s projects will
be deemed leaderless. In this case the TC will oversee PTL selection as be deemed leaderless. In this case the TC will oversee PTL selection as
described by [3]. described by [3].
Thank you, Thank you,
[1] http://governance.openstack.org/election/#how-to-submit-a-candidacy [1] http://governance.openstack.org/election/#how-to-submit-a-candidacy
[2] Assuming the open reviews below are validated [2] Any open reviews at
https://review.openstack.org/#/q/is:open+project:openstack/election https://review.openstack.org/#/q/is:open+project:openstack/election
[3] %s""" have not been factored into these stats.
[3] %(leaderless_url)s""" # noqa
print(email_text % (num_projects_without_candidates, LEADERLESS_URL)) print(email_text % (ptl_fmt_args))
def ptl_end_nominations(projects_no_candidates, def ptl_end_nominations(projects_no_candidates,
@ -465,6 +477,7 @@ def main():
parser_ptl = cmd_parsers.add_parser('ptl') parser_ptl = cmd_parsers.add_parser('ptl')
parser_ptl.add_argument('template', parser_ptl.add_argument('template',
choices=['election_season', 'nominations_kickoff', choices=['election_season', 'nominations_kickoff',
'nominations_last_days',
]) ])
parser_tc = cmd_parsers.add_parser('tc') parser_tc = cmd_parsers.add_parser('tc')
parser_tc.add_argument('template', parser_tc.add_argument('template',