blob: b872d32f2d809d5de1bc174f4ef0c5082a4d28f5 [file] [log] [blame]
koder aka kdanilov4643fd62015-02-10 16:20:13 -08001import re
2import os
3import time
koder aka kdanilove21d7472015-02-14 19:02:04 -08004import logging
koder aka kdanilov4643fd62015-02-10 16:20:13 -08005
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -08006from concurrent.futures import ThreadPoolExecutor
7
koder aka kdanilov652cd802015-04-13 12:21:07 +03008# from novaclient.exceptions import NotFound
koder aka kdanilov4643fd62015-02-10 16:20:13 -08009from novaclient.client import Client as n_client
10from cinderclient.v1.client import Client as c_client
11
koder aka kdanilovda45e882015-04-06 02:24:42 +030012from nodes.node import Node
koder aka kdanilov1c2b5112015-04-10 16:53:51 +030013
koder aka kdanilov4643fd62015-02-10 16:20:13 -080014
koder aka kdanilove21d7472015-02-14 19:02:04 -080015logger = logging.getLogger("io-perf-tool")
16
17
koder aka kdanilov4643fd62015-02-10 16:20:13 -080018def ostack_get_creds():
19 env = os.environ.get
20 name = env('OS_USERNAME')
21 passwd = env('OS_PASSWORD')
22 tenant = env('OS_TENANT_NAME')
23 auth_url = env('OS_AUTH_URL')
gstepanov023c1e42015-04-08 15:50:19 +030024
koder aka kdanilov4643fd62015-02-10 16:20:13 -080025 return name, passwd, tenant, auth_url
26
27
koder aka kdanilov1c2b5112015-04-10 16:53:51 +030028NOVA_CONNECTION = None
29
30
31def nova_connect(name=None, passwd=None, tenant=None, auth_url=None):
32 global NOVA_CONNECTION
33 if NOVA_CONNECTION is None:
34 if name is None:
35 name, passwd, tenant, auth_url = ostack_get_creds()
36 NOVA_CONNECTION = n_client('1.1', name, passwd, tenant, auth_url)
37 return NOVA_CONNECTION
38
39
40def nova_disconnect():
41 global NOVA_CONNECTION
42 if NOVA_CONNECTION is not None:
43 NOVA_CONNECTION.close()
44 NOVA_CONNECTION = None
koder aka kdanilov4643fd62015-02-10 16:20:13 -080045
46
koder aka kdanilov652cd802015-04-13 12:21:07 +030047# def get_or_create_aa_group(nova, name):
48# try:
49# group = conn.server_groups.find(name=name)
50# except NotFound:
51# group = None
52
53# if group is None:
54# conn.server_groups.create
55
56
57def allow_ssh(nova):
58 secgroup = nova.security_groups.find(name="default")
59 nova.security_group_rules.create(secgroup.id,
60 ip_protocol="tcp",
61 from_port="22",
62 to_port="22",
63 cidr="0.0.0.0/0")
64
65 nova.security_group_rules.create(secgroup.id,
66 ip_protocol="icmp",
67 from_port=-1,
68 cidr="0.0.0.0/0",
69 to_port=-1)
70
71
koder aka kdanilov4643fd62015-02-10 16:20:13 -080072def create_keypair(nova, name, key_path):
73 with open(key_path) as key:
74 return nova.keypairs.create(name, key.read())
75
76
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -080077def create_volume(size, name):
koder aka kdanilov4643fd62015-02-10 16:20:13 -080078 cinder = c_client(*ostack_get_creds())
koder aka kdanilov4643fd62015-02-10 16:20:13 -080079 vol = cinder.volumes.create(size=size, display_name=name)
80 err_count = 0
81 while vol.status != 'available':
82 if vol.status == 'error':
83 if err_count == 3:
koder aka kdanilove21d7472015-02-14 19:02:04 -080084 logger.critical("Fail to create volume")
koder aka kdanilov4643fd62015-02-10 16:20:13 -080085 raise RuntimeError("Fail to create volume")
86 else:
87 err_count += 1
88 cinder.volumes.delete(vol)
89 time.sleep(1)
90 vol = cinder.volumes.create(size=size, display_name=name)
91 continue
92 time.sleep(1)
93 vol = cinder.volumes.get(vol.id)
94 return vol
95
96
97def wait_for_server_active(nova, server, timeout=240):
98 t = time.time()
99 while True:
koder aka kdanilov3f356262015-02-13 08:06:14 -0800100 time.sleep(1)
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800101 sstate = getattr(server, 'OS-EXT-STS:vm_state').lower()
102
103 if sstate == 'active':
104 return True
105
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800106 if sstate == 'error':
107 return False
108
109 if time.time() - t > timeout:
110 return False
111
112 server = nova.servers.get(server)
113
114
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800115class Allocate(object):
116 pass
117
118
119def get_floating_ips(nova, pool, amount):
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800120 ip_list = nova.floating_ips.list()
121
122 if pool is not None:
123 ip_list = [ip for ip in ip_list if ip.pool == pool]
124
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800125 return [ip for ip in ip_list if ip.instance_id is None][:amount]
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800126
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800127
koder aka kdanilovda45e882015-04-06 02:24:42 +0300128def launch_vms(config):
koder aka kdanilovda45e882015-04-06 02:24:42 +0300129 logger.debug("Starting new nodes on openstack")
koder aka kdanilovda45e882015-04-06 02:24:42 +0300130 params = config['vm_params'].copy()
131 count = params.pop('count')
132
133 if isinstance(count, basestring):
134 assert count.startswith("x")
koder aka kdanilov1c2b5112015-04-10 16:53:51 +0300135 lst = NOVA_CONNECTION.services.list(binary='nova-compute')
koder aka kdanilovda45e882015-04-06 02:24:42 +0300136 srv_count = len([srv for srv in lst if srv.status == 'enabled'])
137 count = srv_count * int(count[1:])
138
koder aka kdanilov66839a92015-04-11 13:22:31 +0300139 msg_templ = "Will start {0} servers with next params: {1}"
140 logger.debug(msg_templ.format(count, ""))
koder aka kdanilov1c2b5112015-04-10 16:53:51 +0300141 # vm_creds = config['vm_params']['creds'] ?????
142 vm_creds = params.pop('creds')
koder aka kdanilovda45e882015-04-06 02:24:42 +0300143
koder aka kdanilov1c2b5112015-04-10 16:53:51 +0300144 for ip, os_node in create_vms_mt(NOVA_CONNECTION, count, **params):
145 yield Node(vm_creds.format(ip), []), os_node.id
koder aka kdanilovda45e882015-04-06 02:24:42 +0300146
147
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800148def create_vms_mt(nova, amount, keypair_name, img_name,
149 flavor_name, vol_sz=None, network_zone_name=None,
koder aka kdanilov7dec9df2015-02-15 21:35:19 -0800150 flt_ip_pool=None, name_templ='ceph-test-{0}',
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800151 scheduler_hints=None):
152
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800153 with ThreadPoolExecutor(max_workers=16) as executor:
koder aka kdanilov97644f92015-02-13 11:11:08 -0800154 if network_zone_name is not None:
155 network_future = executor.submit(nova.networks.find,
156 label=network_zone_name)
157 else:
158 network_future = None
159
160 fl_future = executor.submit(nova.flavors.find, name=flavor_name)
161 img_future = executor.submit(nova.images.find, name=img_name)
162
163 if flt_ip_pool is not None:
164 ips_future = executor.submit(get_floating_ips,
165 nova, flt_ip_pool, amount)
koder aka kdanilove21d7472015-02-14 19:02:04 -0800166 logger.debug("Wait for floating ip")
koder aka kdanilov97644f92015-02-13 11:11:08 -0800167 ips = ips_future.result()
168 ips += [Allocate] * (amount - len(ips))
169 else:
170 ips = [None] * amount
171
koder aka kdanilov7dec9df2015-02-15 21:35:19 -0800172 logger.debug("Getting flavor object")
koder aka kdanilov97644f92015-02-13 11:11:08 -0800173 fl = fl_future.result()
koder aka kdanilov7dec9df2015-02-15 21:35:19 -0800174 logger.debug("Getting image object")
koder aka kdanilov97644f92015-02-13 11:11:08 -0800175 img = img_future.result()
176
177 if network_future is not None:
koder aka kdanilove21d7472015-02-14 19:02:04 -0800178 logger.debug("Waiting for network results")
koder aka kdanilov97644f92015-02-13 11:11:08 -0800179 nics = [{'net-id': network_future.result().id}]
180 else:
181 nics = None
182
koder aka kdanilov97644f92015-02-13 11:11:08 -0800183 names = map(name_templ.format, range(amount))
184
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800185 futures = []
koder aka kdanilov6e2ae792015-03-04 18:02:24 -0800186 logger.debug("Requesting new vms")
187
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800188 for name, flt_ip in zip(names, ips):
189 params = (nova, name, keypair_name, img, fl,
190 nics, vol_sz, flt_ip, scheduler_hints,
191 flt_ip_pool)
192
193 futures.append(executor.submit(create_vm, *params))
koder aka kdanilove21d7472015-02-14 19:02:04 -0800194 res = [future.result() for future in futures]
195 logger.debug("Done spawning")
196 return res
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800197
198
199def create_vm(nova, name, keypair_name, img,
200 fl, nics, vol_sz=None,
201 flt_ip=False,
202 scheduler_hints=None,
203 pool=None):
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800204 for i in range(3):
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800205 srv = nova.servers.create(name,
206 flavor=fl, image=img, nics=nics,
207 key_name=keypair_name,
208 scheduler_hints=scheduler_hints)
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800209
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800210 if not wait_for_server_active(nova, srv):
211 msg = "Server {0} fails to start. Kill it and try again"
koder aka kdanilove21d7472015-02-14 19:02:04 -0800212 logger.debug(msg.format(srv))
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800213 nova.servers.delete(srv)
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800214
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800215 while True:
koder aka kdanilov3f356262015-02-13 08:06:14 -0800216 # print "wait till server deleted"
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800217 all_id = set(alive_srv.id for alive_srv in nova.servers.list())
218 if srv.id not in all_id:
219 break
220 time.sleep(1)
221 else:
222 break
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800223
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800224 if vol_sz is not None:
koder aka kdanilov3f356262015-02-13 08:06:14 -0800225 # print "creating volume"
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800226 vol = create_volume(vol_sz, name)
koder aka kdanilov3f356262015-02-13 08:06:14 -0800227 # print "attach volume to server"
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800228 nova.volumes.create_server_volume(srv.id, vol.id, None)
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800229
230 if flt_ip is Allocate:
231 flt_ip = nova.floating_ips.create(pool)
koder aka kdanilovda45e882015-04-06 02:24:42 +0300232
koder aka kdanilov7acd6bd2015-02-12 14:28:30 -0800233 if flt_ip is not None:
koder aka kdanilov3f356262015-02-13 08:06:14 -0800234 # print "attaching ip to server"
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800235 srv.add_floating_ip(flt_ip)
Yulia Portnova0e64ea22015-03-20 17:27:22 +0200236
koder aka kdanilovda45e882015-04-06 02:24:42 +0300237 return flt_ip.ip, nova.servers.get(srv.id)
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800238
239
koder aka kdanilov1c2b5112015-04-10 16:53:51 +0300240def clear_nodes(nodes_ids):
241 clear_all(NOVA_CONNECTION, nodes_ids, None)
gstepanov023c1e42015-04-08 15:50:19 +0300242
243
koder aka kdanilov1c2b5112015-04-10 16:53:51 +0300244def clear_all(nova, ids=None, name_templ="ceph-test-{0}"):
245
246 def need_delete(srv):
247 if name_templ is not None:
248 return re.match(name_templ.format("\\d+"), srv.name) is not None
249 else:
250 return srv.id in ids
251
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800252 deleted_srvs = set()
253 for srv in nova.servers.list():
koder aka kdanilov1c2b5112015-04-10 16:53:51 +0300254 if need_delete(srv):
koder aka kdanilove21d7472015-02-14 19:02:04 -0800255 logger.debug("Deleting server {0}".format(srv.name))
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800256 nova.servers.delete(srv)
257 deleted_srvs.add(srv.id)
258
259 while deleted_srvs != set():
koder aka kdanilove21d7472015-02-14 19:02:04 -0800260 logger.debug("Waiting till all servers are actually deleted")
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800261 all_id = set(srv.id for srv in nova.servers.list())
262 if all_id.intersection(deleted_srvs) == set():
koder aka kdanilove21d7472015-02-14 19:02:04 -0800263 logger.debug("Done, deleting volumes")
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800264 break
265 time.sleep(1)
266
267 # wait till vm actually deleted
268
koder aka kdanilov1c2b5112015-04-10 16:53:51 +0300269 if name_templ is not None:
270 cinder = c_client(*ostack_get_creds())
271 for vol in cinder.volumes.list():
272 if isinstance(vol.display_name, basestring):
273 if re.match(name_templ.format("\\d+"), vol.display_name):
274 if vol.status in ('available', 'error'):
275 logger.debug("Deleting volume " + vol.display_name)
276 cinder.volumes.delete(vol)
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800277
koder aka kdanilove21d7472015-02-14 19:02:04 -0800278 logger.debug("Clearing done (yet some volumes may still deleting)")
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800279
280
koder aka kdanilov4643fd62015-02-10 16:20:13 -0800281# def prepare_host(key_file, ip, fio_path, dst_fio_path, user='cirros'):
282# print "Wait till ssh ready...."
283# wait_ssh_ready(ip, user, key_file)
284
285# print "Preparing host >"
286# print " Coping fio"
287# copy_fio(key_file, ip, fio_path, user, dst_fio_path)
288
289# key_opts = '-o UserKnownHostsFile=/dev/null -o StrictHostKeyChecking=no'
290# args = (key_file, user, ip, key_opts)
291# cmd_format = "ssh {3} -i {0} {1}@{2} '{{0}}'".format(*args).format
292
293# def exec_on_host(cmd):
294# print " " + cmd
295# subprocess.check_call(cmd_format(cmd), shell=True)
296
297# exec_on_host("sudo /usr/sbin/mkfs.ext4 /dev/vdb")
298# exec_on_host("sudo /bin/mkdir /media/ceph")
299# exec_on_host("sudo /bin/mount /dev/vdb /media/ceph")
300# exec_on_host("sudo /bin/chmod a+rwx /media/ceph")