Delete update_users.py that no longer needed
Delete update_users.py since this script is not used any more. Use update_gerrit_users.py instead. Change-Id: Ie72b565df058a34e044921288013498b5be9477e Reviewed-on: https://review.openstack.org/12487 Reviewed-by: James E. Blair <corvus@inaugust.com> Approved: Clark Boylan <clark.boylan@gmail.com> Reviewed-by: Clark Boylan <clark.boylan@gmail.com> Tested-by: Jenkins
This commit is contained in:
parent
b689183635
commit
86c0d79588
@ -1,431 +0,0 @@
|
|||||||
#! /usr/bin/env python
|
|
||||||
# Copyright (C) 2012 OpenStack, LLC.
|
|
||||||
#
|
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
# you may not use this file except in compliance with the License.
|
|
||||||
# You may obtain a copy of the License at
|
|
||||||
#
|
|
||||||
# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
#
|
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
||||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
||||||
# License for the specific language governing permissions and limitations
|
|
||||||
# under the License.
|
|
||||||
|
|
||||||
# Synchronize Gerrit users from Launchpad.
|
|
||||||
# TODO items:
|
|
||||||
# 1. add a temporary (instance level) object store for the launchpad class
|
|
||||||
# 2. split out the two classes into separate files to be used as a library
|
|
||||||
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
import ConfigParser
|
|
||||||
import StringIO
|
|
||||||
import paramiko
|
|
||||||
import json
|
|
||||||
import logging
|
|
||||||
import uuid
|
|
||||||
from launchpadlib.launchpad import Launchpad
|
|
||||||
from launchpadlib.uris import LPNET_SERVICE_ROOT
|
|
||||||
|
|
||||||
from datetime import datetime
|
|
||||||
|
|
||||||
from openid.consumer import consumer
|
|
||||||
from openid.cryptutil import randomString
|
|
||||||
|
|
||||||
GERRIT_USER = os.environ.get('GERRIT_USER', 'launchpadsync')
|
|
||||||
GERRIT_CONFIG = os.environ.get('GERRIT_CONFIG',
|
|
||||||
'/home/gerrit2/review_site/etc/gerrit.config')
|
|
||||||
GERRIT_SECURE_CONFIG = os.environ.get('GERRIT_SECURE_CONFIG',
|
|
||||||
'/home/gerrit2/review_site/etc/secure.config')
|
|
||||||
GERRIT_SSH_KEY = os.environ.get('GERRIT_SSH_KEY',
|
|
||||||
'/home/gerrit2/.ssh/launchpadsync_rsa')
|
|
||||||
GERRIT_CACHE_DIR = os.path.expanduser(os.environ.get('GERRIT_CACHE_DIR',
|
|
||||||
'~/.launchpadlib/cache'))
|
|
||||||
GERRIT_CREDENTIALS = os.path.expanduser(os.environ.get('GERRIT_CREDENTIALS',
|
|
||||||
'~/.launchpadlib/creds'))
|
|
||||||
GERRIT_BACKUP_PATH = os.environ.get('GERRIT_BACKUP_PATH',
|
|
||||||
'/home/gerrit2/dbupdates')
|
|
||||||
|
|
||||||
logging.basicConfig(format='%(asctime)-6s: %(name)s - %(levelname)s - %(message)s', filename='/var/log/gerrit/update_users.log')
|
|
||||||
logger= logging.getLogger('update_users')
|
|
||||||
logger.setLevel(logging.INFO)
|
|
||||||
|
|
||||||
for check_path in (os.path.dirname(GERRIT_CACHE_DIR),
|
|
||||||
os.path.dirname(GERRIT_CREDENTIALS),
|
|
||||||
GERRIT_BACKUP_PATH):
|
|
||||||
if not os.path.exists(check_path):
|
|
||||||
os.makedirs(check_path)
|
|
||||||
|
|
||||||
def get_broken_config(filename):
|
|
||||||
""" gerrit config ini files are broken and have leading tabs """
|
|
||||||
text = ""
|
|
||||||
with open(filename,"r") as conf:
|
|
||||||
for line in conf.readlines():
|
|
||||||
text = "%s%s" % (text, line.lstrip())
|
|
||||||
|
|
||||||
fp = StringIO.StringIO(text)
|
|
||||||
c=ConfigParser.ConfigParser()
|
|
||||||
c.readfp(fp)
|
|
||||||
return c
|
|
||||||
|
|
||||||
gerrit_config = get_broken_config(GERRIT_CONFIG)
|
|
||||||
secure_config = get_broken_config(GERRIT_SECURE_CONFIG)
|
|
||||||
|
|
||||||
DB_USER = gerrit_config.get("database", "username")
|
|
||||||
DB_PASS = secure_config.get("database","password")
|
|
||||||
DB_DB = gerrit_config.get("database","database")
|
|
||||||
|
|
||||||
def make_db_backup():
|
|
||||||
db_backup_file = "%s.%s.sql" % (DB_DB, datetime.isoformat(datetime.now()))
|
|
||||||
db_backup_path = os.path.join(GERRIT_BACKUP_PATH, db_backup_file)
|
|
||||||
retval = os.system("mysqldump --opt -u%s -p%s %s | gzip -9 > %s.gz" %
|
|
||||||
(DB_USER, DB_PASS, DB_DB, db_backup_path))
|
|
||||||
if retval != 0:
|
|
||||||
logger.error("Problem taking a db dump, aborting db update")
|
|
||||||
sys.exit(retval)
|
|
||||||
|
|
||||||
class LaunchpadAction(object):
|
|
||||||
def __init__(self):
|
|
||||||
logger.info('Connecting to Launchpad')
|
|
||||||
self.launchpad= Launchpad.login_with('Gerrit User Sync', LPNET_SERVICE_ROOT,
|
|
||||||
GERRIT_CACHE_DIR,
|
|
||||||
credentials_file = GERRIT_CREDENTIALS)
|
|
||||||
|
|
||||||
logger.info('Getting Launchpad teams')
|
|
||||||
self.lp_teams= self.get_all_sub_teams('openstack', [])
|
|
||||||
|
|
||||||
def get_all_sub_teams(self, team, have_teams):
|
|
||||||
for sub_team in self.launchpad.people[team].sub_teams:
|
|
||||||
if sub_team.name not in have_teams:
|
|
||||||
have_teams = self.get_all_sub_teams(sub_team.name, have_teams)
|
|
||||||
have_teams.append(team)
|
|
||||||
return have_teams
|
|
||||||
|
|
||||||
def get_sub_teams(self, team):
|
|
||||||
sub_teams= []
|
|
||||||
for sub_team in self.launchpad.people[team].sub_teams:
|
|
||||||
sub_teams.append(sub_team.name)
|
|
||||||
return sub_teams
|
|
||||||
|
|
||||||
def get_teams(self):
|
|
||||||
return self.lp_teams
|
|
||||||
|
|
||||||
def get_all_users(self):
|
|
||||||
logger.info('Getting Launchpad users')
|
|
||||||
users= []
|
|
||||||
for team in self.lp_teams:
|
|
||||||
for detail in self.launchpad.people[team].members_details:
|
|
||||||
if (detail.status == 'Approved' or detail.status == 'Administrator'):
|
|
||||||
name= detail.self_link.split('/')[-1]
|
|
||||||
if ((users.count(name) == 0) and (name not in self.lp_teams)):
|
|
||||||
users.append(name)
|
|
||||||
return users
|
|
||||||
|
|
||||||
def get_user_data(self, user):
|
|
||||||
return self.launchpad.people[user]
|
|
||||||
|
|
||||||
def get_team_members(self, team, gerrit):
|
|
||||||
users= []
|
|
||||||
for detail in self.launchpad.people[team].members_details:
|
|
||||||
if (detail.status == 'Approved' or detail.status == 'Administrator'):
|
|
||||||
name= detail.self_link.split('/')[-1]
|
|
||||||
# if we found a subteam
|
|
||||||
if name in self.lp_teams:
|
|
||||||
# check subteam for implied subteams
|
|
||||||
for implied_group in gerrit.get_implied_groups(name):
|
|
||||||
if implied_group in self.lp_teams:
|
|
||||||
users.extend(self.get_team_members(implied_group, gerrit))
|
|
||||||
users.extend(self.get_team_members(name, gerrit))
|
|
||||||
continue
|
|
||||||
users.append(name)
|
|
||||||
# check team for implied teams
|
|
||||||
for implied_group in gerrit.get_implied_groups(team):
|
|
||||||
if implied_group in self.lp_teams:
|
|
||||||
users.extend(self.get_team_members(implied_group, gerrit))
|
|
||||||
# filter out dupes
|
|
||||||
users= list(set(users))
|
|
||||||
return users
|
|
||||||
|
|
||||||
def get_team_watches(self, team):
|
|
||||||
users= []
|
|
||||||
for detail in self.launchpad.people[team].members_details:
|
|
||||||
if (detail.status == 'Approved' or detail.status == 'Administrator'):
|
|
||||||
name= detail.self_link.split('/')[-1]
|
|
||||||
if name in self.lp_teams:
|
|
||||||
continue
|
|
||||||
if users.count(name) == 0:
|
|
||||||
users.append(name)
|
|
||||||
return users
|
|
||||||
|
|
||||||
def get_team_display_name(self, team):
|
|
||||||
team_data = self.launchpad.people[team]
|
|
||||||
return team_data.display_name
|
|
||||||
|
|
||||||
class GerritAction(object):
|
|
||||||
def __init__(self):
|
|
||||||
logger.info('Connecting to Gerrit')
|
|
||||||
self.ssh= paramiko.SSHClient()
|
|
||||||
self.ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
|
|
||||||
self.ssh.connect('localhost', username=GERRIT_USER, port=29418, key_filename=GERRIT_SSH_KEY)
|
|
||||||
|
|
||||||
def cleanup(self):
|
|
||||||
logger.info('Closing connection to Gerrit')
|
|
||||||
self.ssh.close()
|
|
||||||
|
|
||||||
def run_query(self, query):
|
|
||||||
command= 'gerrit gsql --format JSON -c "{0}"'.format(query)
|
|
||||||
stdin, stdout, stderr= self.ssh.exec_command(command)
|
|
||||||
# trying to get stdout return code or stderr can hang with large result sets
|
|
||||||
# for line in stderr:
|
|
||||||
# logger.error(line)
|
|
||||||
return stdout
|
|
||||||
|
|
||||||
def get_groups(self):
|
|
||||||
logger.info('Getting Gerrit groups')
|
|
||||||
groups= []
|
|
||||||
query= "select name from account_groups"
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
for line in stdout:
|
|
||||||
row= json.loads(line)
|
|
||||||
if row['type'] == 'row':
|
|
||||||
group= row['columns']['name']
|
|
||||||
groups.append(group)
|
|
||||||
return groups
|
|
||||||
|
|
||||||
def get_users(self):
|
|
||||||
logger.info('Getting Gerrit users')
|
|
||||||
users= []
|
|
||||||
query= "select external_id from account_external_ids"
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
for line in stdout:
|
|
||||||
row= json.loads(line)
|
|
||||||
if row['type'] == 'row':
|
|
||||||
user= row['columns']['external_id'].replace('username:','')
|
|
||||||
users.append(user)
|
|
||||||
return users
|
|
||||||
|
|
||||||
def get_group_id(self, group_name):
|
|
||||||
query= "select group_id from account_groups where name='{0}'".format(group_name)
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
line= stdout.readline()
|
|
||||||
row= json.loads(line)
|
|
||||||
if row['type'] == 'row':
|
|
||||||
return row['columns']['group_id']
|
|
||||||
else:
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def get_user_id(self, user_name):
|
|
||||||
query= "select account_id from account_external_ids where external_id='username:{0}'".format(user_name)
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
line= stdout.readline()
|
|
||||||
row= json.loads(line)
|
|
||||||
return row['columns']['account_id']
|
|
||||||
|
|
||||||
def get_users_from_group(self, group_name):
|
|
||||||
logger.info('Getting Gerrit users from group %s', group_name)
|
|
||||||
users= []
|
|
||||||
gid= self.get_group_id(group_name)
|
|
||||||
|
|
||||||
query= "select external_id from account_external_ids join account_group_members on account_group_members.account_id=account_external_ids.account_id where account_group_members.group_id={0} and external_id like 'username%%'".format(gid)
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
for line in stdout:
|
|
||||||
row= json.loads(line)
|
|
||||||
if row['type'] == 'row':
|
|
||||||
user= row['columns']['external_id'].replace('username:','')
|
|
||||||
users.append(user)
|
|
||||||
return users
|
|
||||||
|
|
||||||
def get_users_from_watches(self, group_name):
|
|
||||||
logger.info('Getting Gerrit users from watch list %s', group_name)
|
|
||||||
users= []
|
|
||||||
if group_name.endswith("-core"):
|
|
||||||
group_name = group_name[:-5]
|
|
||||||
group_name = "openstack/{0}".format(group_name)
|
|
||||||
|
|
||||||
query= "select external_id from account_external_ids join account_project_watches on account_project_watches.account_id=account_external_ids.account_id where account_project_watches.project_name like '{0}' and external_id like 'username%%'".format(group_name)
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
for line in stdout:
|
|
||||||
row= json.loads(line)
|
|
||||||
if row['type'] == 'row':
|
|
||||||
user= row['columns']['external_id'].replace('username:','')
|
|
||||||
users.append(user)
|
|
||||||
return users
|
|
||||||
|
|
||||||
|
|
||||||
def get_implied_groups(self, group_name):
|
|
||||||
gid= self.get_group_id(group_name)
|
|
||||||
groups= []
|
|
||||||
query= "select name from account_groups join account_group_includes on account_group_includes.include_id=account_groups.group_id where account_group_includes.group_id={0}".format(gid)
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
for line in stdout:
|
|
||||||
row= json.loads(line)
|
|
||||||
if row['type'] == 'row':
|
|
||||||
group= row['columns']['name']
|
|
||||||
groups.append(group)
|
|
||||||
return groups
|
|
||||||
|
|
||||||
def add_group(self, group_name, group_display_name):
|
|
||||||
logger.info('New group %s (%s)', group_display_name, group)
|
|
||||||
query= "insert into account_group_id (s) values (NULL)"
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
row= json.loads(stdout.readline())
|
|
||||||
if row['rowCount'] is not 1:
|
|
||||||
print "Could not get a new account group ID"
|
|
||||||
raise
|
|
||||||
query= "select max(s) from account_group_id"
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
row= json.loads(stdout.readline())
|
|
||||||
gid= row['columns']['max(s)']
|
|
||||||
full_uuid= "{0}{1}".format(uuid.uuid4().hex, uuid.uuid4().hex[:8])
|
|
||||||
query= "insert into account_groups (group_id, group_type, owner_group_id, name, description, group_uuid) values ({0}, 'INTERNAL', 1, '{1}', '{2}', '{3}')". format(gid, group_name, group_display_name, full_uuid)
|
|
||||||
self.run_query(query)
|
|
||||||
query= "insert into account_group_names (group_id, name) values ({0}, '{1}')".format(gid, group_name)
|
|
||||||
self.run_query(query)
|
|
||||||
|
|
||||||
def add_user(self, user_name, user_data):
|
|
||||||
logger.info("Adding Gerrit user %s", user_name)
|
|
||||||
openid_consumer = consumer.Consumer(dict(id=randomString(16, '0123456789abcdef')), None)
|
|
||||||
openid_request = openid_consumer.begin("https://launchpad.net/~%s" % user_data.name)
|
|
||||||
user_openid_external_id = openid_request.endpoint.getLocalID()
|
|
||||||
query= "select account_id from account_external_ids where external_id in ('{0}')".format(user_openid_external_id)
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
row= json.loads(stdout.readline())
|
|
||||||
if row['type'] == 'row':
|
|
||||||
# we have a result so this is an updated user name
|
|
||||||
account_id= row['columns']['account_id']
|
|
||||||
query= "update account_external_ids set external_id='{0}' where external_id like 'username%%' and account_id = {1}".format('username:%s' % user_name, account_id)
|
|
||||||
self.run_query(query)
|
|
||||||
else:
|
|
||||||
# we really do have a new user
|
|
||||||
user_ssh_keys= ["%s %s %s" % ('ssh-%s' % key.keytype.lower(), key.keytext, key.comment) for key in user_data.sshkeys]
|
|
||||||
user_email= None
|
|
||||||
try:
|
|
||||||
user_email = user_data.preferred_email_address.email
|
|
||||||
except ValueError:
|
|
||||||
pass
|
|
||||||
query= "insert into account_id (s) values (NULL)"
|
|
||||||
self.run_query(query)
|
|
||||||
query= "select max(s) from account_id"
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
row= json.loads(stdout.readline())
|
|
||||||
uid= row['columns']['max(s)']
|
|
||||||
query= "insert into accounts (account_id, full_name, preferred_email) values ({0}, '{1}', '{2}')".format(uid, user_name, user_email)
|
|
||||||
self.run_query(query)
|
|
||||||
keyno= 1
|
|
||||||
for key in user_ssh_keys:
|
|
||||||
query= "insert into account_ssh_keys (ssh_public_key, valid, account_id, seq) values ('{0}', 'Y', {1}, {2})".format(key.strip(), uid, keyno)
|
|
||||||
self.run_query(query)
|
|
||||||
keyno = keyno + 1
|
|
||||||
query= "insert into account_external_ids (account_id, email_address, external_id) values ({0}, '{1}', '{2}')".format(uid, user_email, user_openid_external_id)
|
|
||||||
self.run_query(query)
|
|
||||||
query= "insert into account_external_ids (account_id, external_id) values ({0}, '{1}')".format(uid, "username:%s" % user_name)
|
|
||||||
self.run_query(query)
|
|
||||||
if user_email is not None:
|
|
||||||
query= "insert into account_external_ids (account_id, email_address, external_id) values ({0}. '{1}', '{2}')".format(uid, user_email, "mailto:%s" % user_email)
|
|
||||||
return None
|
|
||||||
|
|
||||||
def add_user_to_group(self, user_name, group_name):
|
|
||||||
logger.info("Adding Gerrit user %s to group %s", user_name, group_name)
|
|
||||||
uid= self.get_user_id(user_name)
|
|
||||||
gid= self.get_group_id(group_name)
|
|
||||||
if gid is 0:
|
|
||||||
print "Trying to add user {0} to non-existent group {1}".format(user_name, group_name)
|
|
||||||
raise
|
|
||||||
query= "insert into account_group_members (account_id, group_id) values ({0}, {1})".format(uid, gid)
|
|
||||||
self.run_query(query)
|
|
||||||
|
|
||||||
def add_user_to_watch(self, user_name, group_name):
|
|
||||||
logger.info("Adding Gerrit user %s to watch group %s", user_name, group_name)
|
|
||||||
uid= self.get_user_id(user_name)
|
|
||||||
if group_name.endswith("-core"):
|
|
||||||
group_name = group_name[:-5]
|
|
||||||
group_name = "openstack/{0}".format(group_name)
|
|
||||||
query= "insert into account_project_watches VALUES ('Y', 'N', 'N', {0}, '{1}', '*')". format(uid, group_name)
|
|
||||||
self.run_query(query)
|
|
||||||
|
|
||||||
|
|
||||||
def del_user_from_group(self, user_name, group_name):
|
|
||||||
logger.info("Deleting Gerrit user %s from group %s", user_name, group_name)
|
|
||||||
uid= self.get_user_id(user_name)
|
|
||||||
gid= self.get_group_id(group_name)
|
|
||||||
query= "delete from account_group_members where account_id = {0} and group_id = {1}".format(uid, gid)
|
|
||||||
self.run_query(query)
|
|
||||||
if group_name.endswith("-core"):
|
|
||||||
group_name = group_name[:-5]
|
|
||||||
group_name= "openstack/{0}".format(group_name)
|
|
||||||
query= "delete from account_project_watches where account_id = {0} and project_name= '{1}'".format(uid, group_name)
|
|
||||||
self.run_query(query)
|
|
||||||
|
|
||||||
def rebuild_sub_groups(self, group, sub_groups):
|
|
||||||
gid= self.get_group_id(group)
|
|
||||||
for sub_group in sub_groups:
|
|
||||||
sgid= self.get_group_id(sub_group)
|
|
||||||
query= "select group_id from account_group_includes where group_id={0} and include_id={1}".format(gid, sgid)
|
|
||||||
stdout= self.run_query(query)
|
|
||||||
row= json.loads(stdout.readline())
|
|
||||||
if row['type'] != 'row':
|
|
||||||
logger.info('Adding implied group %s to group %s', group, sub_group)
|
|
||||||
query= "insert into account_group_includes (group_id, include_id) values ({0}, {1})".format(gid, sgid)
|
|
||||||
self.run_query(query)
|
|
||||||
|
|
||||||
|
|
||||||
# Actual work starts here!
|
|
||||||
|
|
||||||
lp= LaunchpadAction()
|
|
||||||
gerrit= GerritAction()
|
|
||||||
|
|
||||||
logger.info('Making DB backup')
|
|
||||||
make_db_backup()
|
|
||||||
|
|
||||||
logger.info('Starting group reconcile')
|
|
||||||
lp_groups= lp.get_teams()
|
|
||||||
gerrit_groups= gerrit.get_groups()
|
|
||||||
|
|
||||||
group_diff= filter(lambda a: a not in gerrit_groups, lp_groups)
|
|
||||||
for group in group_diff:
|
|
||||||
group_display_name= lp.get_team_display_name(group)
|
|
||||||
gerrit.add_group(group, group_display_name)
|
|
||||||
|
|
||||||
for group in lp_groups:
|
|
||||||
sub_group= lp.get_sub_teams(group)
|
|
||||||
if sub_group:
|
|
||||||
gerrit.rebuild_sub_groups(group, sub_group)
|
|
||||||
|
|
||||||
logger.info('End group reconcile')
|
|
||||||
|
|
||||||
logger.info('Starting user reconcile')
|
|
||||||
lp_users= lp.get_all_users()
|
|
||||||
gerrit_users= gerrit.get_users()
|
|
||||||
|
|
||||||
user_diff= filter(lambda a: a not in gerrit_users, lp_users)
|
|
||||||
for user in user_diff:
|
|
||||||
gerrit.add_user(user, lp.get_user_data(user))
|
|
||||||
|
|
||||||
logger.info('End user reconcile')
|
|
||||||
|
|
||||||
logger.info('Starting user to group reconcile')
|
|
||||||
lp_groups= lp.get_teams()
|
|
||||||
for group in lp_groups:
|
|
||||||
# First find users to attach to groups
|
|
||||||
gerrit_group_users= gerrit.get_users_from_group(group)
|
|
||||||
lp_group_users= lp.get_team_members(group, gerrit)
|
|
||||||
|
|
||||||
group_diff= filter(lambda a: a not in gerrit_group_users, lp_group_users)
|
|
||||||
for user in group_diff:
|
|
||||||
gerrit.add_user_to_group(user, group)
|
|
||||||
# Second find users to attach to watches
|
|
||||||
lp_group_watches= lp.get_team_watches(group)
|
|
||||||
gerrit_group_watches= gerrit.get_users_from_watches(group)
|
|
||||||
group_diff= filter(lambda a: a not in gerrit_group_watches, lp_group_watches)
|
|
||||||
for user in group_diff:
|
|
||||||
gerrit.add_user_to_watch(user, group)
|
|
||||||
# Third find users to remove from groups/watches
|
|
||||||
group_diff= filter(lambda a: a not in lp_group_users, gerrit_group_users)
|
|
||||||
for user in group_diff:
|
|
||||||
gerrit.del_user_from_group(user, group)
|
|
||||||
|
|
||||||
logger.info('Ending user to group reconcile')
|
|
||||||
|
|
||||||
gerrit.cleanup()
|
|
Loading…
Reference in New Issue
Block a user