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
|
# Copyright (C) 2015 Codethink Limited
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; version 2 of the License.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License along
# with this program; if not, write to the Free Software Foundation, Inc.,
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
import logging
import cliapp
from . import hosts
class GerritDownstream(hosts.DownstreamHost):
'''Run commands on a Gerrit instance.
This uses the SSH API to Gerrit. The REST API is actually much nicer to
use, but it requires that the account doing stuff has set an HTTP password.
Use of the SSH API requires only the SSH key that we also use for push
access.
'''
@staticmethod
def check_app_settings(app_settings):
if app_settings['downstream-visibility'] != 'private':
raise cliapp.ApplicationError(
'Cannot create non-private repositories in Gerrit')
def __init__(self, app_settings):
# XXX These need to be configurable
host = 'localhost'
port = 29418
user = 'lorry'
self._ssh_command_args = [
'ssh', '-oStrictHostKeyChecking=no', '-oBatchMode=yes', '-p%i' % port,
'%s@%s' % (user, host)]
def _ssh_command(self, command):
out = cliapp.runcmd(self._ssh_command_args + command)
if isinstance(out, bytes):
out = out.decode('utf-8', errors='replace')
return out
def _has_project(self, name):
# There's no 'does this project exist' command in Gerrit 2.9.4; 'list
# all projects with this prefix' is as close we can get.
output = self._ssh_command([
'gerrit', 'ls-projects', '--type=ALL', '--prefix=%s' % name])
projects = output.strip().split('\n')
if name in projects:
return True
else:
return False
def prepare_repo(self, name, metadata):
'''Create a project in the local Gerrit server.
The 'lorry' user must have createProject capability in the Gerrit.
'''
if self._has_project(name):
logging.info('Project %s exists in local Gerrit already.',
name)
else:
self._ssh_command(['gerrit', 'create-project', name])
logging.info('Created %s project in local Gerrit.', name)
# We can only set this metadata if we're the owner of the
# repository. For now, ignore failures.
try:
if 'head' in metadata:
self._ssh_command(['gerrit', 'set-head', name,
'--new-head', metadata['head']])
if 'description' in metadata:
self._ssh_command(['gerrit', 'set-project', name,
'-d', metadata['description']])
except cliapp.AppException:
pass
|