From aae968f2af7deb121774bf5142c2a6a0cd79369e Mon Sep 17 00:00:00 2001 From: Ghanshyam Mann Date: Wed, 1 Feb 2023 19:00:01 -0600 Subject: [PATCH] Migrate election tooling to release versions From Antelope (2023.1) onwards, primary identifier of any release is release version instead of name[1], we should use release versions in the election tooling also. This migrate the election tooling to use the release version As release repo is not fully migrated to release version yet, we still need to use the release name to fetch the data from release repo. Keeping relaese version as float so that we edit the configuration file with release version a float not with string even string is passed. All the election tools/script/flow has been tested with it and it is working now. [1] https://governance.openstack.org/tc/reference/release-naming.html Change-Id: Ia8a60a9514b1dca5ac97ed8099c34c6c6c7d24e2 --- .../cmds/setup_election_config.py | 44 ++++++++++++++----- openstack_election/cmds/template_emails.py | 2 +- openstack_election/config.py | 1 + 3 files changed, 36 insertions(+), 11 deletions(-) diff --git a/openstack_election/cmds/setup_election_config.py b/openstack_election/cmds/setup_election_config.py index b0763699..6f249bb2 100755 --- a/openstack_election/cmds/setup_election_config.py +++ b/openstack_election/cmds/setup_election_config.py @@ -45,6 +45,16 @@ def _dict_representer(dumper, data): return dumper.represent_dict(data.items()) +def valid_version(value): + try: + value = float(value) + except ValueError: + msg = ("{} is not an valid release version. Pass release version, for" + " example 2023.1".format(value)) + raise Exception(msg) + return value + + def valid_date(opt): try: d = datetime.datetime.strptime(opt, "%Y-%m-%d") @@ -110,7 +120,8 @@ def main(): parser = argparse.ArgumentParser(description=('Given a release ' 'date pick some dates for ' 'the election')) - parser.add_argument('release', help='release name') + parser.add_argument('release', type=valid_version, + help='release version. Example 2023.2') parser.add_argument('type', choices=['TC', 'PTL', 'combined']) parser.add_argument('--tc-seats', default=4, choices=['4', '5'], help='number of TC seats up for election') @@ -121,7 +132,6 @@ def main(): 'YYYY-MM-DD') args = parser.parse_args() - args.release = args.release.lower() params = election_parameters[args.type] offset = datetime.timedelta(weeks=params['weeks_prior']) @@ -129,11 +139,25 @@ def main(): # We need to know the releases in order. Fortunately this data exists # in the releases repo in a really easy format to understand. series_data = utils.get_series_data() - names = [x['name'].lower() for x in series_data] + # TODO(gmann): release versions (release-id) is introduced from 2023.1 + # (Antelope) cycle and before that we have only release 'name'. We fetch + # last 3 release data here so until 2023.1 we can fecth release-ids and + # rest with 'name'. + # We will use release version for election tooling and release repo is not + # yet fully moved to the release version so continue using the release + # name to fetch the data from release repo. + release_data = [] + names = [] + for x in series_data: + names.append(x['name']) + if 'release-id' in x: + release_data.append(str(x['release-id'])) + else: + release_data.append(x['name']) # Find where in the list the requested release sits. This will typically # be very early but don't assume that. - idx = names.index(args.release) if args.release in names else -1 - + idx = (release_data.index(str(args.release)) if str(args.release) + in release_data else -1) # If date is not passed to this script then autimatically # select the release end date. if (args.date is None): @@ -147,8 +171,8 @@ def main(): # Stein, Rocky[0], Queens[1], Pike[2], Ocata[3] # For the Stein elections candidates come from the previous 2 cycles so: # Rocky and Queens. - timeframe_name = '%s-%s' % (names[idx+2].capitalize(), - names[idx+1].capitalize()) + timeframe_name = '%s-%s' % (release_data[idx+2].capitalize(), + release_data[idx+1].capitalize()) # The Queens development cycle begins when we branch Pike-RC1, so collect # that date from the releases data. @@ -222,9 +246,9 @@ def main(): timeframe_end = email_deadline print('Setting TC timeframe end to email_deadline') - print('Begining of %s Cycle @ %s' % (names[idx+2].capitalize(), + print('Begining of %s Cycle @ %s' % (release_data[idx+2].capitalize(), timeframe_start)) - print('End of %s cycle @ %s' % (names[idx+1].capitalize(), + print('End of %s cycle @ %s' % (release_data[idx+1].capitalize(), timeframe_end)) timeframe_span = timeframe_end - timeframe_start @@ -240,7 +264,7 @@ def main(): print('Maximum: %s' % (datetime.timedelta(13*365/12))) configuration = OrderedDict( - release=args.release.lower(), + release=args.release, election_type=args.type.lower(), tag=args.date.strftime('%b-%Y-elections').lower(), tc_seats=int(args.tc_seats), diff --git a/openstack_election/cmds/template_emails.py b/openstack_election/cmds/template_emails.py index 215ef403..1e6dd8d8 100644 --- a/openstack_election/cmds/template_emails.py +++ b/openstack_election/cmds/template_emails.py @@ -43,7 +43,7 @@ if election_type in ['tc', 'combined']: election_start=utils.get_event('TC Election')['start_str'], election_end=utils.get_event('TC Election')['end_str'], poll_name='OpenStack %s Cycle Technical Committee Election Poll' % ( - conf['release'].capitalize()), + conf['release']), )) template_names += ['campaigning_kickoff'] diff --git a/openstack_election/config.py b/openstack_election/config.py index 203d3a0d..3e925666 100644 --- a/openstack_election/config.py +++ b/openstack_election/config.py @@ -27,6 +27,7 @@ def parse_datetime(iso_format): def load_conf(): conf = yaml.safe_load(open('configuration.yaml')) + conf['release'] = str(conf['release']) for key in ('start', 'end', 'email_deadline'): date = parse_datetime(conf['timeframe'][key]) conf['timeframe'][key] = date