blob: 6e61c55a4db1accfb8f20a149536a909b16152e7 [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
Matthew Treinish96e9e882014-06-09 18:37:19 -040017import six
18
Masayuki Igawa7e9eb542014-02-17 15:03:44 +090019from tempest.common import ssh
Sean Dague86bd8422013-12-20 09:56:44 -050020from tempest import config
Masayuki Igawa7e9eb542014-02-17 15:03:44 +090021from tempest import exceptions
Daryl Walleck98e66dd2012-06-21 04:58:39 -050022
Sean Dague86bd8422013-12-20 09:56:44 -050023CONF = config.CONF
24
Daryl Walleck6b9b2882012-04-08 21:43:39 -050025
26class RemoteClient():
27
Attila Fazekasc3a095b2013-08-17 09:15:44 +020028 # NOTE(afazekas): It should always get an address instead of server
Attila Fazekasa23f5002012-10-23 19:32:45 +020029 def __init__(self, server, username, password=None, pkey=None):
Sean Dague86bd8422013-12-20 09:56:44 -050030 ssh_timeout = CONF.compute.ssh_timeout
31 network = CONF.compute.network_for_ssh
32 ip_version = CONF.compute.ip_version_for_ssh
33 ssh_channel_timeout = CONF.compute.ssh_channel_timeout
llg821243b20502014-02-22 10:32:49 +080034 if isinstance(server, six.string_types):
Attila Fazekasa23f5002012-10-23 19:32:45 +020035 ip_address = server
36 else:
37 addresses = server['addresses'][network]
38 for address in addresses:
39 if address['version'] == ip_version:
40 ip_address = address['addr']
41 break
42 else:
Masayuki Igawa7e9eb542014-02-17 15:03:44 +090043 raise exceptions.ServerUnreachable()
44 self.ssh_client = ssh.Client(ip_address, username, password,
45 ssh_timeout, pkey=pkey,
46 channel_timeout=ssh_channel_timeout)
Daryl Walleck6b9b2882012-04-08 21:43:39 -050047
Elena Ezhova91db24e2014-02-28 20:47:10 +040048 def exec_command(self, cmd):
49 return self.ssh_client.exec_command(cmd)
50
Attila Fazekasad7ef7d2013-11-20 10:12:53 +010051 def validate_authentication(self):
52 """Validate ssh connection and authentication
53 This method raises an Exception when the validation fails.
54 """
55 self.ssh_client.test_connection_auth()
Daryl Walleck6b9b2882012-04-08 21:43:39 -050056
57 def hostname_equals_servername(self, expected_hostname):
Chang Bo Guocc1623c2013-09-13 20:11:27 -070058 # Get host name using command "hostname"
Elena Ezhova91db24e2014-02-28 20:47:10 +040059 actual_hostname = self.exec_command("hostname").rstrip()
Daryl Walleck6b9b2882012-04-08 21:43:39 -050060 return expected_hostname == actual_hostname
61
Daryl Walleck6b9b2882012-04-08 21:43:39 -050062 def get_ram_size_in_mb(self):
Elena Ezhova91db24e2014-02-28 20:47:10 +040063 output = self.exec_command('free -m | grep Mem')
Daryl Walleck6b9b2882012-04-08 21:43:39 -050064 if output:
65 return output.split()[1]
66
67 def get_number_of_vcpus(self):
68 command = 'cat /proc/cpuinfo | grep processor | wc -l'
Elena Ezhova91db24e2014-02-28 20:47:10 +040069 output = self.exec_command(command)
Daryl Walleck6b9b2882012-04-08 21:43:39 -050070 return int(output)
Dan Smithc18d8c62012-07-02 08:09:26 -070071
72 def get_partitions(self):
73 # Return the contents of /proc/partitions
74 command = 'cat /proc/partitions'
Elena Ezhova91db24e2014-02-28 20:47:10 +040075 output = self.exec_command(command)
Dan Smithc18d8c62012-07-02 08:09:26 -070076 return output
Daryl Walleck98e66dd2012-06-21 04:58:39 -050077
78 def get_boot_time(self):
Vincent Untz3c0b5b92014-01-18 10:56:00 +010079 cmd = 'cut -f1 -d. /proc/uptime'
Elena Ezhova91db24e2014-02-28 20:47:10 +040080 boot_secs = self.exec_command(cmd)
Vincent Untz3c0b5b92014-01-18 10:56:00 +010081 boot_time = time.time() - int(boot_secs)
82 return time.localtime(boot_time)
Attila Fazekasa23f5002012-10-23 19:32:45 +020083
84 def write_to_console(self, message):
85 message = re.sub("([$\\`])", "\\\\\\\\\\1", message)
86 # usually to /dev/ttyS0
87 cmd = 'sudo sh -c "echo \\"%s\\" >/dev/console"' % message
Elena Ezhova91db24e2014-02-28 20:47:10 +040088 return self.exec_command(cmd)
Yair Fried5f670ab2013-12-09 09:26:51 +020089
90 def ping_host(self, host):
Kirill Shileev14113572014-11-21 16:58:02 +030091 addr = netaddr.IPAddress(host)
92 cmd = 'ping6' if addr.version == 6 else 'ping'
93 cmd += ' -c1 -w1 {0}'.format(host)
Elena Ezhova91db24e2014-02-28 20:47:10 +040094 return self.exec_command(cmd)
Yair Fried4d7efa62013-11-17 17:12:29 +020095
96 def get_mac_address(self):
Shang Yong1b95cf72014-10-15 17:19:48 +080097 cmd = "/bin/ip addr | awk '/ether/ {print $2}'"
Elena Ezhova91db24e2014-02-28 20:47:10 +040098 return self.exec_command(cmd)
Yair Fried3097dc12014-01-26 08:46:43 +020099
Yair Fried413bf2d2014-11-19 17:07:11 +0200100 def get_nic_name(self, address):
101 cmd = "/bin/ip -o addr | awk '/%s/ {print $2}'" % address
102 return self.exec_command(cmd)
103
Yair Fried3097dc12014-01-26 08:46:43 +0200104 def get_ip_list(self):
105 cmd = "/bin/ip address"
Elena Ezhova91db24e2014-02-28 20:47:10 +0400106 return self.exec_command(cmd)
Yair Fried3097dc12014-01-26 08:46:43 +0200107
108 def assign_static_ip(self, nic, addr):
109 cmd = "sudo /bin/ip addr add {ip}/{mask} dev {nic}".format(
110 ip=addr, mask=CONF.network.tenant_network_mask_bits,
111 nic=nic
112 )
Elena Ezhova91db24e2014-02-28 20:47:10 +0400113 return self.exec_command(cmd)
Yair Fried3097dc12014-01-26 08:46:43 +0200114
115 def turn_nic_on(self, nic):
116 cmd = "sudo /bin/ip link set {nic} up".format(nic=nic)
Elena Ezhova91db24e2014-02-28 20:47:10 +0400117 return self.exec_command(cmd)
Elena Ezhova4a27b462014-04-09 15:25:46 +0400118
119 def get_pids(self, pr_name):
120 # Get pid(s) of a process/program
121 cmd = "ps -ef | grep %s | grep -v 'grep' | awk {'print $1'}" % pr_name
122 return self.exec_command(cmd).split('\n')
Yair Fried413bf2d2014-11-19 17:07:11 +0200123
124 def get_dns_servers(self):
125 cmd = 'cat /etc/resolv.conf'
126 resolve_file = self.exec_command(cmd).strip().split('\n')
127 entries = (l.split() for l in resolve_file)
128 dns_servers = [l[1] for l in entries
129 if len(l) and l[0] == 'nameserver']
130 return dns_servers
131
132 def send_signal(self, pid, signum):
133 cmd = 'sudo /bin/kill -{sig} {pid}'.format(pid=pid, sig=signum)
134 return self.exec_command(cmd)
135
136 def _renew_lease_udhcpc(self, fixed_ip=None):
137 """Renews DHCP lease via udhcpc client. """
138 file_path = '/var/run/udhcpc.'
139 nic_name = self.get_nic_name(fixed_ip)
140 nic_name = nic_name.strip().lower()
141 pid = self.exec_command('cat {path}{nic}.pid'.
142 format(path=file_path, nic=nic_name))
143 pid = pid.strip()
144 self.send_signal(pid, 'USR1')
145
146 def _renew_lease_dhclient(self, fixed_ip=None):
147 """Renews DHCP lease via dhclient client. """
148 cmd = "sudo /sbin/dhclient -r && /sbin/dhclient"
149 self.exec_command(cmd)
150
151 def renew_lease(self, fixed_ip=None):
152 """Wrapper method for renewing DHCP lease via given client
153
154 Supporting:
155 * udhcpc
156 * dhclient
157 """
158 # TODO(yfried): add support for dhcpcd
159 suported_clients = ['udhcpc', 'dhclient']
160 dhcp_client = CONF.scenario.dhcp_client
161 if dhcp_client not in suported_clients:
162 raise exceptions.InvalidConfiguration('%s DHCP client unsupported'
163 % dhcp_client)
164 if dhcp_client == 'udhcpc' and not fixed_ip:
165 raise ValueError("need to set 'fixed_ip' for udhcpc client")
166 return getattr(self, '_renew_lease_' + dhcp_client)(fixed_ip=fixed_ip)