Add SSH connectivity to overcloud nodes.

Change-Id: Ib87d31a47860d1fa9589d5fe3b08010f99e5f338
This commit is contained in:
Federico Ressi 2019-08-27 13:16:48 +02:00
parent 54cfaa673e
commit 96fbbab05e
11 changed files with 261 additions and 105 deletions

View File

@ -0,0 +1,4 @@
{
"context_is_admin": "role:dummy",
"context_is_advsvc": "role:dummy"
}

View File

@ -0,0 +1,8 @@
[DEFAULT]
# Show debugging output in logs (sets DEBUG log level output)
debug = False
lock_path = $state_path/lock
[database]
connection = 'sqlite://'

View File

@ -0,0 +1,5 @@
{
"context_is_admin": "role:admin",
"context_is_advsvc": "role:advsvc",
"default": "rule:admin_or_owner"
}

View File

@ -26,3 +26,4 @@ ssh_client = _client.ssh_client
ssh_command = _command.ssh_command
ssh_proxy_client = _client.ssh_proxy_client
SSHConnectFailure = _client.SSHConnectFailure
gather_ssh_connect_parameters = _client.gather_ssh_connect_parameters

View File

@ -15,6 +15,8 @@
# under the License.
from __future__ import absolute_import
import collections
import getpass
import os
import socket
@ -31,12 +33,48 @@ from tobiko.shell.ssh import _command
LOG = log.getLogger(__name__)
def valid_hostname(value):
hostname = str(value)
if not hostname:
message = "Invalid hostname: {!r}".format(hostname)
raise ValueError(message)
return hostname
def valid_port(value):
port = int(value)
if port <= 0 or port > 65535:
message = "Invalid port number: {!r}".format(port)
raise ValueError(message)
return port
def valid_path(value):
return os.path.abspath(os.path.expanduser(value))
def positive_float(value):
value = float(value)
if value <= 0.:
message = "{!r} is not positive".format(value)
raise ValueError(message)
return value
def positive_int(value):
value = int(value)
if value <= 0:
message = "{!r} is not positive".format(value)
raise ValueError(message)
return value
SSH_CONNECT_PARAMETERS = {
#: The server to connect to
'hostname': str,
'hostname': valid_hostname,
#: The server port to connect to
'port': int,
'port': valid_port,
#: The username to authenticate as (defaults to the current local username)
'username': str,
@ -50,10 +88,10 @@ SSH_CONNECT_PARAMETERS = {
#: The filename, or list of filenames, of optional private key(s) and/or
#: certs to try for authentication
'key_filename': str,
'key_filename': os.path.expanduser,
#: An optional timeout (in seconds) for the TCP connect
'timeout': float,
'timeout': positive_float,
#: Set to False to disable connecting to the SSH agent
'allow_agent': bool,
@ -83,22 +121,77 @@ SSH_CONNECT_PARAMETERS = {
#: An optional timeout (in seconds) to wait for the SSH banner to be
#: presented.
'banner_timeout': float,
'banner_timeout': positive_float,
#: An optional timeout (in seconds) to wait for an authentication response
'auth_timeout': float,
'auth_timeout': positive_float,
#: Number of connection attempts to be tried before timeout
'connection_attempts': int,
'connection_attempts': positive_int,
#: Minimum amount of time to wait between two connection attempts
'connection_interval': float,
'connection_interval': positive_float,
#: Command to be executed to open proxy sock
'proxy_command': str,
}
def gather_ssh_connect_parameters(source=None, destination=None, schema=None,
remove_from_schema=False, **kwargs):
if schema is None:
assert not remove_from_schema
schema = SSH_CONNECT_PARAMETERS
parameters = {}
if source:
# gather from object
if isinstance(source, collections.Mapping):
parameters.update(_items_from_mapping(mapping=source,
schema=schema))
else:
parameters.update(_items_from_object(obj=source,
schema=schema))
if kwargs:
# gather from kwargs
parameters.update(_items_from_mapping(mapping=kwargs, schema=schema))
kwargs = exclude_mapping_items(kwargs, schema)
if kwargs:
message = "Invalid SSH connection parameters: {!r}".format(kwargs)
raise ValueError(message)
if remove_from_schema and parameters:
# update schema
for name in parameters:
del schema[name]
if destination is not None:
destination.update(parameters)
return parameters
def _items_from_mapping(mapping, schema):
for name, init in schema.items():
value = mapping.get(name)
if value is not None:
yield name, init(value)
def _items_from_object(obj, schema):
for name, init in schema.items():
value = getattr(obj, name, None)
if value is not None:
yield name, init(value)
def exclude_mapping_items(mapping, exclude):
# Exclude parameters that are already in target dictionary
return {key: value
for key, value in mapping.items()
if key not in exclude}
class SSHConnectFailure(tobiko.TobikoException):
message = "Failed to login to {login}\n{cause}"
@ -118,9 +211,10 @@ class SSHClientFixture(tobiko.SharedFixture):
proxy_client = None
proxy_sock = None
connect_parameters = None
schema = SSH_CONNECT_PARAMETERS
def __init__(self, host=None, proxy_client=None, host_config=None,
config_files=None, **connect_parameters):
config_files=None, schema=None, **kwargs):
super(SSHClientFixture, self).__init__()
if host:
self.host = host
@ -130,14 +224,10 @@ class SSHClientFixture(tobiko.SharedFixture):
self.host_config = host_config
if config_files:
self.config_files = config_files
invalid_parameters = sorted([name
for name in connect_parameters
if name not in SSH_CONNECT_PARAMETERS])
if invalid_parameters:
message = "Invalid SSH connection parameters: {!s}".format(
', '.join(invalid_parameters))
raise ValueError(message)
self._connect_parameters = connect_parameters
self.schema = schema = dict(schema or self.schema)
self._connect_parameters = gather_ssh_connect_parameters(
schema=schema, **kwargs)
def setup_fixture(self):
self.setup_host_config()
@ -157,71 +247,49 @@ class SSHClientFixture(tobiko.SharedFixture):
- parameters got from ~/.ssh/config and tobiko.conf
- parameters got from fixture object attributes
"""
self.connect_parameters = self.get_connect_parameters()
# Get default parameter values from self object
self.connect_parameters = parameters = items_from_object(
schema=SSH_CONNECT_PARAMETERS, obj=self)
LOG.debug('Default parameters for host %r:\n%r', self.host,
def get_connect_parameters(self, schema=None):
schema = dict(schema or self.schema)
parameters = {}
for gather_parameters in [self.gather_initial_connect_parameters,
self.gather_host_config_connect_parameters,
self.gather_default_connect_parameters]:
gather_parameters(destination=parameters,
schema=schema,
remove_from_schema=True)
if parameters:
LOG.debug('SSH connect parameters for host %r:\n%r', self.host,
parameters)
return parameters
# Override parameters from host configuration files
parameters.update(
items_from_mapping(schema=SSH_CONNECT_PARAMETERS,
mapping=self.host_config.connect_parameters))
LOG.debug('Configured connect parameters for host %r:\n%r',
self.host, parameters)
def gather_initial_connect_parameters(self, **kwargs):
parameters = gather_ssh_connect_parameters(
source=self._connect_parameters, **kwargs)
if parameters:
LOG.debug('Initial SSH connect parameters for host %r:\n'
'%r', self.host, parameters)
return parameters
# Override parameters with __init__ parameters
parameters.update(
items_from_mapping(schema=SSH_CONNECT_PARAMETERS,
mapping=self._connect_parameters))
LOG.debug('Resulting connect parameters for host %r:\n%r', self.host,
parameters)
def gather_host_config_connect_parameters(self, **kwargs):
parameters = gather_ssh_connect_parameters(
source=self.host_config.connect_parameters, **kwargs)
if parameters:
LOG.debug('Host configured SSH connect parameters for host %r:\n'
'%r', self.host, parameters)
return parameters
# Validate hostname
hostname = parameters.get('hostname')
if not hostname:
message = "Invalid hostname: {!r}".format(hostname)
raise ValueError(message)
# Expand key_filename
key_filename = parameters.get('key_filename')
if key_filename:
key_filename = os.path.expanduser(key_filename)
if not os.path.exists(key_filename):
message = "key_filename {!r} doesn't exist".format(hostname)
raise ValueError(message)
parameters['key_filename'] = key_filename
# Validate connection timeout
timeout = parameters.get('timeout')
if not timeout or timeout < 0.:
message = "Invalid timeout: {!r}".format(timeout)
raise ValueError(message)
# Validate connection attempts
connection_attempts = parameters.get('connection_attempts')
if not connection_attempts or connection_attempts < 0:
message = "Invalid connection attempts: {!r}".format(
connection_attempts)
raise ValueError(message)
# Validate connection attempts
connection_interval = parameters.get('connection_interval')
if not connection_interval or connection_interval < 0.:
message = "Invalid connection interval: {!r}".format(
connection_interval)
raise ValueError(message)
# Validate connection port
port = parameters.get('port')
if not port or port < 1 or port > 65535:
message = "Invalid port: {!r}".format(port)
raise ValueError(message)
def gather_default_connect_parameters(self, **kwargs):
parameters = gather_ssh_connect_parameters(source=self, **kwargs)
if parameters:
LOG.debug('Default SSH connect parameters for host %r:\n'
'%r', self.host, parameters)
return parameters
def setup_ssh_client(self):
self.client, self.proxy_sock = ssh_connect(
proxy_client=self.proxy_client, **self.connect_parameters)
proxy_client=self.proxy_client,
**self.connect_parameters)
self.addCleanup(self.client.close)
if self.proxy_sock:
self.addCleanup(self.proxy_sock.close)
@ -230,18 +298,6 @@ class SSHClientFixture(tobiko.SharedFixture):
return tobiko.setup_fixture(self).client
def items_from_mapping(schema, mapping):
return ((key, init(mapping.get(key)))
for key, init in schema.items()
if mapping.get(key) is not None)
def items_from_object(schema, obj):
return {key: init(getattr(obj, key, None))
for key, init in schema.items()
if getattr(obj, key, None) is not None}
UNDEFINED_CLIENT = 'UNDEFINED_CLIENT'
@ -269,7 +325,8 @@ class SSHClientManager(object):
config_files=config_files)
self.clients[host_key] = client = SSHClientFixture(
host=host, hostname=hostname, port=port, username=username,
proxy_client=proxy_client, **connect_parameters)
proxy_client=proxy_client, host_config=host_config,
**connect_parameters)
return client
def get_proxy_client(self, host=None, proxy_jump=None, host_config=None,

View File

@ -13,6 +13,8 @@
# under the License.
from __future__ import absolute_import
import os
import netaddr
import testtools
@ -61,13 +63,22 @@ class OvercloudNovaApiTest(testtools.TestCase):
self.assertIsInstance(overcloud_node_ip, netaddr.IPAddress)
def test_overcloud_host_config(self):
hostname = overcloud.find_overcloud_node().name
host_config = tobiko.setup_fixture(
overcloud.overcloud_host_config(hostname='controller-0'))
self.assertEqual('controller-0', host_config.host)
overcloud.overcloud_host_config(hostname=hostname))
self.assertEqual(hostname, host_config.host)
self.assertIsInstance(host_config.hostname, netaddr.IPAddress)
self.assertEqual(CONF.tobiko.tripleo.overcloud_ssh_port,
host_config.port)
self.assertEqual(CONF.tobiko.tripleo.overcloud_ssh_username,
host_config.username)
self.assertEqual(CONF.tobiko.tripleo.ssh_key_filename,
host_config.key_filename)
key_filename = os.path.expanduser(
CONF.tobiko.tripleo.overcloud_ssh_key_filename)
self.assertEqual(key_filename, host_config.key_filename)
self.assertTrue(os.path.isfile(key_filename))
self.assertTrue(os.path.isfile(key_filename + '.pub'))
def test_overcloud_ssh_client_connection(self):
hostname = overcloud.find_overcloud_node().name
ssh_client = overcloud.overcloud_ssh_client(hostname=hostname)
ssh_client.connect()

View File

@ -26,7 +26,7 @@ TIPLEO_CONF = CONF.tobiko.tripleo
class TripleoConfigTest(unit.TobikoUnitTest):
def test_ssh_key_filename(self):
self.assertIsInstance(TIPLEO_CONF.ssh_key_filename,
self.assertIsInstance(TIPLEO_CONF.undercloud_ssh_key_filename,
six.string_types)

View File

@ -19,11 +19,6 @@ from oslo_config import cfg
GROUP_NAME = 'tripleo'
OPTIONS = [
# TripleO options
cfg.StrOpt('ssh_key_filename',
default='~/.ssh/id_rsa',
help="SSH key filename used to login to TripleO nodes"),
# Undercloud options
cfg.StrOpt('undercloud_ssh_hostname',
default=None,
@ -35,6 +30,9 @@ OPTIONS = [
cfg.StrOpt('undercloud_ssh_username',
default='stack',
help="Username with access to stackrc and overcloudrc files"),
cfg.StrOpt('undercloud_ssh_key_filename',
default='~/.ssh/id_rsa',
help="SSH key filename used to login to Undercloud node"),
cfg.StrOpt('undercloud_rcfile',
default='~/stackrc',
help="Undercloud RC filename"),
@ -46,6 +44,9 @@ OPTIONS = [
cfg.StrOpt('overcloud_ssh_username',
default='heat-admin',
help="Default username used to connect to overcloud nodes"),
cfg.StrOpt('overcloud_ssh_key_filename',
default='~/.ssh/id_overcloud',
help="SSH key filename used to login to Overcloud nodes"),
cfg.StrOpt('overcloud_rcfile',
default='~/overcloudrc',
help="Overcloud RC filename"),

View File

@ -13,12 +13,17 @@
# under the License.
from __future__ import absolute_import
import io
import os
import six
import tobiko
from tobiko import config
from tobiko.openstack import keystone
from tobiko.openstack import nova
from tobiko.shell import sh
from tobiko.shell import ssh
from tobiko.tripleo import undercloud
@ -56,6 +61,13 @@ def find_overcloud_node(**params):
return nova.find_server(client=client, **params)
def overcloud_ssh_client(hostname, ip_version=None, network_name=None):
host_config = overcloud_host_config(hostname=hostname,
ip_version=ip_version,
network_name=network_name)
return ssh.ssh_client(host=hostname, host_config=host_config)
def overcloud_host_config(hostname, ip_version=None, network_name=None):
host_config = OvercloudHostConfig(host=hostname,
ip_version=ip_version,
@ -72,15 +84,52 @@ def overcloud_node_ip_address(ip_version=None, network_name=None,
network_name=network_name)
class OvercloudSshKeyFileFixture(tobiko.SharedFixture):
key_filename = os.path.expanduser(
CONF.tobiko.tripleo.overcloud_ssh_key_filename)
def setup_fixture(self):
key_filename = self.key_filename
if not os.path.isfile(key_filename):
self.setup_key_file()
assert os.path.isfile(key_filename)
def setup_key_file(self):
key_filename = self.key_filename
key_dirname = os.path.dirname(key_filename)
tobiko.makedirs(key_dirname, mode=0o700)
ssh_client = undercloud.undercloud_ssh_client()
_get_undercloud_file(ssh_client=ssh_client,
source='~/.ssh/id_rsa',
destination=key_filename,
mode=0o600)
_get_undercloud_file(ssh_client=ssh_client,
source='~/.ssh/id_rsa.pub',
destination=key_filename + '.pub',
mode=0o600)
def _get_undercloud_file(ssh_client, source, destination, mode):
content = sh.execute(['cat', source],
ssh_client=ssh_client).stdout
with io.open(destination, 'wb') as fd:
fd.write(content.encode())
os.chmod(destination, mode)
class OvercloudHostConfig(tobiko.SharedFixture):
hostname = None
port = None
username = None
key_filename = None
key_file = tobiko.required_setup_fixture(OvercloudSshKeyFileFixture)
ip_version = None
network_name = None
key_filename = None
def __init__(self, host, ip_version=None, network_name=None):
def __init__(self, host, ip_version=None, network_name=None,
**kwargs):
super(OvercloudHostConfig, self).__init__()
tobiko.check_valid_type(host, six.string_types)
self.host = host
@ -88,11 +137,19 @@ class OvercloudHostConfig(tobiko.SharedFixture):
self.ip_version = ip_version
if network_name:
self.network_name = network_name
self._connect_parameters = ssh.gather_ssh_connect_parameters(**kwargs)
def setup_fixture(self):
self.hostname = overcloud_node_ip_address(
self.hostname = self.hostname or overcloud_node_ip_address(
name=self.host, ip_version=self.ip_version,
network_name=self.network_name)
self.port = CONF.tobiko.tripleo.overcloud_ssh_port
self.username = CONF.tobiko.tripleo.overcloud_ssh_username
self.key_filename = CONF.tobiko.tripleo.ssh_key_filename
self.port = self.port or CONF.tobiko.tripleo.overcloud_ssh_port
self.username = (self.username or
CONF.tobiko.tripleo.overcloud_ssh_username)
self.key_filename = self.key_filename or self.key_file.key_filename
@property
def connect_parameters(self):
parameters = ssh.gather_ssh_connect_parameters(self)
parameters.update(self._connect_parameters)
return parameters

View File

@ -68,11 +68,21 @@ class UndecloudHostConfig(tobiko.SharedFixture):
username = None
key_filename = None
def __init__(self, **kwargs):
super(UndecloudHostConfig, self).__init__()
self._connect_parameters = ssh.gather_ssh_connect_parameters(**kwargs)
def setup_fixture(self):
self.hostname = CONF.tobiko.tripleo.undercloud_ssh_hostname
self.port = CONF.tobiko.tripleo.undercloud_ssh_port
self.username = CONF.tobiko.tripleo.undercloud_ssh_username
self.key_filename = CONF.tobiko.tripleo.ssh_key_filename
self.key_filename = CONF.tobiko.tripleo.undercloud_ssh_key_filename
@property
def connect_parameters(self):
parameters = ssh.gather_ssh_connect_parameters(self)
parameters.update(self._connect_parameters)
return parameters
def undercloud_keystone_client():