Merge "Allow to query Gerrit behind corporate proxy"
This commit is contained in:
commit
7eaea71b6b
@ -169,6 +169,10 @@
|
|||||||
|
|
||||||
# SSH username for gerrit review system access (string value)
|
# SSH username for gerrit review system access (string value)
|
||||||
#ssh_username = user
|
#ssh_username = user
|
||||||
|
#
|
||||||
|
# Proxy command to be used by ssh transport to query Gerrit behind corporate proxy
|
||||||
|
# E.g. for http proxy: "/usr/bin/ncat --proxy-type http --proxy <proxy_host>:<proxy_port> %s %s"
|
||||||
|
#proxy_command = <None>
|
||||||
|
|
||||||
# URI of translation team data (string value)
|
# URI of translation team data (string value)
|
||||||
#translation_team_uri = https://opendev.org/openstack/i18n/raw/branch/master/tools/zanata/translation_team.yaml
|
#translation_team_uri = https://opendev.org/openstack/i18n/raw/branch/master/tools/zanata/translation_team.yaml
|
||||||
|
@ -63,6 +63,8 @@ PROCESSOR_OPTS = [
|
|||||||
help='Number of seconds to wait for remote response'),
|
help='Number of seconds to wait for remote response'),
|
||||||
cfg.IntOpt('gerrit-retry', default=10,
|
cfg.IntOpt('gerrit-retry', default=10,
|
||||||
help='How many times to retry after Gerrit errors'),
|
help='How many times to retry after Gerrit errors'),
|
||||||
|
cfg.StrOpt('proxy-command', default=None,
|
||||||
|
help='Proxy command to support Gerrit query behind proxy'),
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
import json
|
import json
|
||||||
import re
|
import re
|
||||||
|
|
||||||
|
from oslo_config import cfg
|
||||||
from oslo_log import log as logging
|
from oslo_log import log as logging
|
||||||
import paramiko
|
import paramiko
|
||||||
import time
|
import time
|
||||||
@ -28,6 +29,8 @@ PAGE_LIMIT = 100
|
|||||||
REQUEST_COUNT_LIMIT = 20
|
REQUEST_COUNT_LIMIT = 20
|
||||||
SSH_ERRORS_LIMIT = 10
|
SSH_ERRORS_LIMIT = 10
|
||||||
|
|
||||||
|
CONF = cfg.CONF
|
||||||
|
|
||||||
|
|
||||||
def get_socket_tuple_from_uri(uri):
|
def get_socket_tuple_from_uri(uri):
|
||||||
stripped = re.sub(GERRIT_URI_PREFIX, '', uri)
|
stripped = re.sub(GERRIT_URI_PREFIX, '', uri)
|
||||||
@ -72,6 +75,11 @@ class Gerrit(Rcs):
|
|||||||
|
|
||||||
self.key_filename = None
|
self.key_filename = None
|
||||||
self.username = None
|
self.username = None
|
||||||
|
self.sock = None
|
||||||
|
self.proxy_command = CONF.proxy_command
|
||||||
|
if self.proxy_command:
|
||||||
|
self.proxy_command %= (self.hostname, self.port)
|
||||||
|
|
||||||
self.ssh_errors_limit = SSH_ERRORS_LIMIT
|
self.ssh_errors_limit = SSH_ERRORS_LIMIT
|
||||||
|
|
||||||
self.client = paramiko.SSHClient()
|
self.client = paramiko.SSHClient()
|
||||||
@ -92,10 +100,12 @@ class Gerrit(Rcs):
|
|||||||
self._connect()
|
self._connect()
|
||||||
|
|
||||||
def _connect(self):
|
def _connect(self):
|
||||||
|
if self.proxy_command:
|
||||||
|
self.sock = paramiko.ProxyCommand(self.proxy_command)
|
||||||
try:
|
try:
|
||||||
self.client.connect(self.hostname, port=self.port,
|
self.client.connect(self.hostname, port=self.port,
|
||||||
key_filename=self.key_filename,
|
key_filename=self.key_filename,
|
||||||
username=self.username)
|
username=self.username, sock=self.sock)
|
||||||
LOG.debug('Successfully connected to Gerrit')
|
LOG.debug('Successfully connected to Gerrit')
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
LOG.error('Failed to connect to gerrit %(host)s:%(port)s. '
|
LOG.error('Failed to connect to gerrit %(host)s:%(port)s. '
|
||||||
@ -210,6 +220,8 @@ class Gerrit(Rcs):
|
|||||||
|
|
||||||
def close(self):
|
def close(self):
|
||||||
self.client.close()
|
self.client.close()
|
||||||
|
if self.sock:
|
||||||
|
self.sock.close()
|
||||||
|
|
||||||
|
|
||||||
def get_rcs(uri):
|
def get_rcs(uri):
|
||||||
|
@ -16,10 +16,14 @@
|
|||||||
import json
|
import json
|
||||||
|
|
||||||
import mock
|
import mock
|
||||||
|
from oslo_config import cfg
|
||||||
import testtools
|
import testtools
|
||||||
|
|
||||||
|
from stackalytics.processor import config
|
||||||
from stackalytics.processor import rcs
|
from stackalytics.processor import rcs
|
||||||
|
|
||||||
|
CONF = cfg.CONF
|
||||||
|
|
||||||
REVIEW_ONE = json.dumps(
|
REVIEW_ONE = json.dumps(
|
||||||
{"project": "openstack/nova", "branch": "master", "topic": "bug/1494374",
|
{"project": "openstack/nova", "branch": "master", "topic": "bug/1494374",
|
||||||
"id": "Id741dfc769c02a5544691a7db49a7dbff6b11376", "number": "229382",
|
"id": "Id741dfc769c02a5544691a7db49a7dbff6b11376", "number": "229382",
|
||||||
@ -32,6 +36,10 @@ REVIEW_END_LINE = json.dumps(
|
|||||||
|
|
||||||
class TestRcs(testtools.TestCase):
|
class TestRcs(testtools.TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestRcs, self).setUp()
|
||||||
|
CONF.register_opts(config.CONNECTION_OPTS + config.PROCESSOR_OPTS)
|
||||||
|
|
||||||
@mock.patch('paramiko.SSHClient')
|
@mock.patch('paramiko.SSHClient')
|
||||||
def test_setup(self, mock_client_cons):
|
def test_setup(self, mock_client_cons):
|
||||||
mock_client = mock.Mock()
|
mock_client = mock.Mock()
|
||||||
@ -45,7 +53,30 @@ class TestRcs(testtools.TestCase):
|
|||||||
|
|
||||||
mock_connect.assert_called_once_with(
|
mock_connect.assert_called_once_with(
|
||||||
'review.openstack.org', port=rcs.DEFAULT_PORT, key_filename='key',
|
'review.openstack.org', port=rcs.DEFAULT_PORT, key_filename='key',
|
||||||
username='user')
|
username='user', sock=None)
|
||||||
|
|
||||||
|
@mock.patch('paramiko.ProxyCommand')
|
||||||
|
@mock.patch('paramiko.SSHClient')
|
||||||
|
def test_setup_with_proxy(self, mock_client_cons, mock_proxy_command):
|
||||||
|
mock_client = mock.Mock()
|
||||||
|
mock_client_cons.return_value = mock_client
|
||||||
|
|
||||||
|
mock_connect = mock.Mock()
|
||||||
|
mock_client.connect = mock_connect
|
||||||
|
|
||||||
|
mock_proxy = mock.Mock()
|
||||||
|
mock_proxy_command.return_value = mock_proxy
|
||||||
|
|
||||||
|
CONF.set_override('proxy_command', '%s:%s')
|
||||||
|
|
||||||
|
gerrit = rcs.Gerrit('gerrit://review.openstack.org')
|
||||||
|
gerrit.setup(username='user', key_filename='key')
|
||||||
|
|
||||||
|
mock_proxy_command.assert_called_once_with(
|
||||||
|
'review.openstack.org:%s' % rcs.DEFAULT_PORT)
|
||||||
|
mock_connect.assert_called_once_with(
|
||||||
|
'review.openstack.org', port=rcs.DEFAULT_PORT, key_filename='key',
|
||||||
|
username='user', sock=mock_proxy)
|
||||||
|
|
||||||
@mock.patch('paramiko.SSHClient')
|
@mock.patch('paramiko.SSHClient')
|
||||||
def test_setup_error(self, mock_client_cons):
|
def test_setup_error(self, mock_client_cons):
|
||||||
@ -62,7 +93,7 @@ class TestRcs(testtools.TestCase):
|
|||||||
|
|
||||||
mock_connect.assert_called_once_with(
|
mock_connect.assert_called_once_with(
|
||||||
'review.openstack.org', port=rcs.DEFAULT_PORT, key_filename='key',
|
'review.openstack.org', port=rcs.DEFAULT_PORT, key_filename='key',
|
||||||
username='user')
|
username='user', sock=None)
|
||||||
|
|
||||||
@mock.patch('paramiko.SSHClient')
|
@mock.patch('paramiko.SSHClient')
|
||||||
@mock.patch('time.time')
|
@mock.patch('time.time')
|
||||||
|
Loading…
Reference in New Issue
Block a user