blob: 6150b4d5fd6ef821f93e61f24164d266a1eb1fff [file] [log] [blame]
Sean Dague556add52013-07-19 14:28:44 -04001# Licensed under the Apache License, Version 2.0 (the "License"); you may
2# not use this file except in compliance with the License. You may obtain
3# a copy of the License at
4#
5# http://www.apache.org/licenses/LICENSE-2.0
6#
7# Unless required by applicable law or agreed to in writing, software
8# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
9# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
10# License for the specific language governing permissions and limitations
11# under the License.
12
Kirill Shileev14113572014-11-21 16:58:02 +030013import netaddr
Attila Fazekasa23f5002012-10-23 19:32:45 +020014import re
Matthew Treinisha83a16e2012-12-07 13:44:02 -050015import time
16
David Kranz968f1b32015-06-18 16:58:18 -040017from oslo_log import log as logging
Matthew Treinish96e9e882014-06-09 18:37:19 -040018import six
Andrey Pavlov64723762015-04-29 06:24:58 +030019from tempest_lib.common import ssh
Matthew Treinish96e9e882014-06-09 18:37:19 -040020
Sean Dague86bd8422013-12-20 09:56:44 -050021from tempest import config
Masayuki Igawa7e9eb542014-02-17 15:03:44 +090022from tempest import exceptions
Daryl Walleck98e66dd2012-06-21 04:58:39 -050023
Sean Dague86bd8422013-12-20 09:56:44 -050024CONF = config.CONF
25
David Kranz968f1b32015-06-18 16:58:18 -040026LOG = logging.getLogger(__name__)
27
Daryl Walleck6b9b2882012-04-08 21:43:39 -050028
Joe Gordon28788b42015-02-25 12:42:37 -080029class RemoteClient(object):
Daryl Walleck6b9b2882012-04-08 21:43:39 -050030
Attila Fazekasc3a095b2013-08-17 09:15:44 +020031 # NOTE(afazekas): It should always get an address instead of server
Attila Fazekasa23f5002012-10-23 19:32:45 +020032 def __init__(self, server, username, password=None, pkey=None):
nithya-ganesan67da2872015-02-08 23:13:48 +000033 ssh_timeout = CONF.validation.ssh_timeout
lanoux283273b2015-12-04 03:01:54 -080034 network = CONF.validation.network_for_ssh
nithya-ganesan67da2872015-02-08 23:13:48 +000035 ip_version = CONF.validation.ip_version_for_ssh
36 connect_timeout = CONF.validation.connect_timeout
llg821243b20502014-02-22 10:32:49 +080037 if isinstance(server, six.string_types):
Attila Fazekasa23f5002012-10-23 19:32:45 +020038 ip_address = server
39 else:
40 addresses = server['addresses'][network]
41 for address in addresses:
42 if address['version'] == ip_version:
43 ip_address = address['addr']
44 break
45 else:
Masayuki Igawa7e9eb542014-02-17 15:03:44 +090046 raise exceptions.ServerUnreachable()
47 self.ssh_client = ssh.Client(ip_address, username, password,
48 ssh_timeout, pkey=pkey,
nithya-ganesan67da2872015-02-08 23:13:48 +000049 channel_timeout=connect_timeout)
Daryl Walleck6b9b2882012-04-08 21:43:39 -050050
Elena Ezhova91db24e2014-02-28 20:47:10 +040051 def exec_command(self, cmd):
Evgeny Antyshevf58ab6d2015-04-15 08:23:05 +000052 # Shell options below add more clearness on failures,
53 # path is extended for some non-cirros guest oses (centos7)
lanoux283273b2015-12-04 03:01:54 -080054 cmd = CONF.validation.ssh_shell_prologue + " " + cmd
David Kranz968f1b32015-06-18 16:58:18 -040055 LOG.debug("Remote command: %s" % cmd)
Elena Ezhova91db24e2014-02-28 20:47:10 +040056 return self.ssh_client.exec_command(cmd)
57
Attila Fazekasad7ef7d2013-11-20 10:12:53 +010058 def validate_authentication(self):
59 """Validate ssh connection and authentication
Ken'ichi Ohmichicb67d2d2015-11-19 08:23:22 +000060
Attila Fazekasad7ef7d2013-11-20 10:12:53 +010061 This method raises an Exception when the validation fails.
62 """
63 self.ssh_client.test_connection_auth()
Daryl Walleck6b9b2882012-04-08 21:43:39 -050064
65 def hostname_equals_servername(self, expected_hostname):
Chang Bo Guocc1623c2013-09-13 20:11:27 -070066 # Get host name using command "hostname"
Elena Ezhova91db24e2014-02-28 20:47:10 +040067 actual_hostname = self.exec_command("hostname").rstrip()
Daryl Walleck6b9b2882012-04-08 21:43:39 -050068 return expected_hostname == actual_hostname
69
Daryl Walleck6b9b2882012-04-08 21:43:39 -050070 def get_ram_size_in_mb(self):
Elena Ezhova91db24e2014-02-28 20:47:10 +040071 output = self.exec_command('free -m | grep Mem')
Daryl Walleck6b9b2882012-04-08 21:43:39 -050072 if output:
73 return output.split()[1]
74
75 def get_number_of_vcpus(self):
Alexander Gubanov0ff3ffb2015-07-01 15:49:06 +030076 output = self.exec_command('grep -c processor /proc/cpuinfo')
Daryl Walleck6b9b2882012-04-08 21:43:39 -050077 return int(output)
Dan Smithc18d8c62012-07-02 08:09:26 -070078
79 def get_partitions(self):
80 # Return the contents of /proc/partitions
81 command = 'cat /proc/partitions'
Elena Ezhova91db24e2014-02-28 20:47:10 +040082 output = self.exec_command(command)
Dan Smithc18d8c62012-07-02 08:09:26 -070083 return output
Daryl Walleck98e66dd2012-06-21 04:58:39 -050084
85 def get_boot_time(self):
Vincent Untz3c0b5b92014-01-18 10:56:00 +010086 cmd = 'cut -f1 -d. /proc/uptime'
Elena Ezhova91db24e2014-02-28 20:47:10 +040087 boot_secs = self.exec_command(cmd)
Vincent Untz3c0b5b92014-01-18 10:56:00 +010088 boot_time = time.time() - int(boot_secs)
89 return time.localtime(boot_time)
Attila Fazekasa23f5002012-10-23 19:32:45 +020090
91 def write_to_console(self, message):
92 message = re.sub("([$\\`])", "\\\\\\\\\\1", message)
93 # usually to /dev/ttyS0
94 cmd = 'sudo sh -c "echo \\"%s\\" >/dev/console"' % message
Elena Ezhova91db24e2014-02-28 20:47:10 +040095 return self.exec_command(cmd)
Yair Fried5f670ab2013-12-09 09:26:51 +020096
lanoux283273b2015-12-04 03:01:54 -080097 def ping_host(self, host, count=CONF.validation.ping_count,
98 size=CONF.validation.ping_size, nic=None):
Kirill Shileev14113572014-11-21 16:58:02 +030099 addr = netaddr.IPAddress(host)
100 cmd = 'ping6' if addr.version == 6 else 'ping'
Yair Friedbc46f592015-11-18 16:29:34 +0200101 if nic:
102 cmd = 'sudo {cmd} -I {nic}'.format(cmd=cmd, nic=nic)
Richard Wintersf87059b2015-02-17 11:46:54 -0500103 cmd += ' -c{0} -w{0} -s{1} {2}'.format(count, size, host)
Elena Ezhova91db24e2014-02-28 20:47:10 +0400104 return self.exec_command(cmd)
Yair Fried4d7efa62013-11-17 17:12:29 +0200105
Yair Friedbc46f592015-11-18 16:29:34 +0200106 def set_mac_address(self, nic, address):
107 self.set_nic_state(nic=nic, state="down")
108 cmd = "sudo ip link set dev {0} address {1}".format(nic, address)
109 self.exec_command(cmd)
110 self.set_nic_state(nic=nic, state="up")
111
112 def get_mac_address(self, nic=""):
113 show_nic = "show {nic} ".format(nic=nic) if nic else ""
114 cmd = "ip addr %s| awk '/ether/ {print $2}'" % show_nic
115 return self.exec_command(cmd).strip().lower()
Yair Fried3097dc12014-01-26 08:46:43 +0200116
Yair Fried413bf2d2014-11-19 17:07:11 +0200117 def get_nic_name(self, address):
Evgeny Antyshevf58ab6d2015-04-15 08:23:05 +0000118 cmd = "ip -o addr | awk '/%s/ {print $2}'" % address
Daniel Mellado9e3e1062015-08-06 18:07:05 +0200119 nic = self.exec_command(cmd)
120 return nic.strip().strip(":").lower()
Yair Fried413bf2d2014-11-19 17:07:11 +0200121
Yair Fried3097dc12014-01-26 08:46:43 +0200122 def get_ip_list(self):
Evgeny Antyshevf58ab6d2015-04-15 08:23:05 +0000123 cmd = "ip address"
Elena Ezhova91db24e2014-02-28 20:47:10 +0400124 return self.exec_command(cmd)
Yair Fried3097dc12014-01-26 08:46:43 +0200125
126 def assign_static_ip(self, nic, addr):
Evgeny Antyshevf58ab6d2015-04-15 08:23:05 +0000127 cmd = "sudo ip addr add {ip}/{mask} dev {nic}".format(
Yair Fried3097dc12014-01-26 08:46:43 +0200128 ip=addr, mask=CONF.network.tenant_network_mask_bits,
129 nic=nic
130 )
Elena Ezhova91db24e2014-02-28 20:47:10 +0400131 return self.exec_command(cmd)
Yair Fried3097dc12014-01-26 08:46:43 +0200132
Yair Friedbc46f592015-11-18 16:29:34 +0200133 def set_nic_state(self, nic, state="up"):
134 cmd = "sudo ip link set {nic} {state}".format(nic=nic, state=state)
Elena Ezhova91db24e2014-02-28 20:47:10 +0400135 return self.exec_command(cmd)
Elena Ezhova4a27b462014-04-09 15:25:46 +0400136
137 def get_pids(self, pr_name):
138 # Get pid(s) of a process/program
139 cmd = "ps -ef | grep %s | grep -v 'grep' | awk {'print $1'}" % pr_name
140 return self.exec_command(cmd).split('\n')
Yair Fried413bf2d2014-11-19 17:07:11 +0200141
142 def get_dns_servers(self):
143 cmd = 'cat /etc/resolv.conf'
144 resolve_file = self.exec_command(cmd).strip().split('\n')
145 entries = (l.split() for l in resolve_file)
146 dns_servers = [l[1] for l in entries
147 if len(l) and l[0] == 'nameserver']
148 return dns_servers
149
150 def send_signal(self, pid, signum):
151 cmd = 'sudo /bin/kill -{sig} {pid}'.format(pid=pid, sig=signum)
152 return self.exec_command(cmd)
153
154 def _renew_lease_udhcpc(self, fixed_ip=None):
155 """Renews DHCP lease via udhcpc client. """
156 file_path = '/var/run/udhcpc.'
157 nic_name = self.get_nic_name(fixed_ip)
Yair Fried413bf2d2014-11-19 17:07:11 +0200158 pid = self.exec_command('cat {path}{nic}.pid'.
159 format(path=file_path, nic=nic_name))
160 pid = pid.strip()
161 self.send_signal(pid, 'USR1')
162
163 def _renew_lease_dhclient(self, fixed_ip=None):
164 """Renews DHCP lease via dhclient client. """
Itzik Brownffb14022015-03-23 17:03:55 +0200165 cmd = "sudo /sbin/dhclient -r && sudo /sbin/dhclient"
Yair Fried413bf2d2014-11-19 17:07:11 +0200166 self.exec_command(cmd)
167
168 def renew_lease(self, fixed_ip=None):
169 """Wrapper method for renewing DHCP lease via given client
170
171 Supporting:
172 * udhcpc
173 * dhclient
174 """
175 # TODO(yfried): add support for dhcpcd
Takashi NATSUME6d5a2b42015-09-08 11:27:49 +0900176 supported_clients = ['udhcpc', 'dhclient']
Yair Fried413bf2d2014-11-19 17:07:11 +0200177 dhcp_client = CONF.scenario.dhcp_client
Takashi NATSUME6d5a2b42015-09-08 11:27:49 +0900178 if dhcp_client not in supported_clients:
Yair Fried413bf2d2014-11-19 17:07:11 +0200179 raise exceptions.InvalidConfiguration('%s DHCP client unsupported'
180 % dhcp_client)
181 if dhcp_client == 'udhcpc' and not fixed_ip:
182 raise ValueError("need to set 'fixed_ip' for udhcpc client")
Joe Gordon28788b42015-02-25 12:42:37 -0800183 return getattr(self, '_renew_lease_' + dhcp_client)(fixed_ip=fixed_ip)
Alexander Gubanovabd154c2015-09-23 23:24:06 +0300184
185 def mount(self, dev_name, mount_path='/mnt'):
186 cmd_mount = 'sudo mount /dev/%s %s' % (dev_name, mount_path)
187 self.exec_command(cmd_mount)
188
189 def umount(self, mount_path='/mnt'):
190 self.exec_command('sudo umount %s' % mount_path)
191
192 def make_fs(self, dev_name, fs='ext4'):
193 cmd_mkfs = 'sudo /usr/sbin/mke2fs -t %s /dev/%s' % (fs, dev_name)
194 self.exec_command(cmd_mkfs)