blob: 07bb6ceab066e5f2655835b66bd39c9bc03b88fa [file] [log] [blame]
ivan-zhu09111942013-08-01 08:09:16 +08001# vim: tabstop=4 shiftwidth=4 softtabstop=4
2
3# Copyright 2012 OpenStack Foundation
4# Copyright 2013 Hewlett-Packard Development Company, L.P.
5# All Rights Reserved.
6#
7# Licensed under the Apache License, Version 2.0 (the "License"); you may
8# not use this file except in compliance with the License. You may obtain
9# a copy of the License at
10#
11# http://www.apache.org/licenses/LICENSE-2.0
12#
13# Unless required by applicable law or agreed to in writing, software
14# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
15# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
16# License for the specific language governing permissions and limitations
17# under the License.
18
19import json
20import time
21import urllib
22
23from tempest.common.rest_client import RestClient
24from tempest.common import waiters
25from tempest import exceptions
26
27
28class ServersClientJSON(RestClient):
29
30 def __init__(self, config, username, password, auth_url, tenant_name=None,
31 auth_version='v2'):
32 super(ServersClientJSON, self).__init__(config, username, password,
33 auth_url, tenant_name,
34 auth_version=auth_version)
35 self.service = self.config.compute.catalog_type
36
37 def create_server(self, name, image_ref, flavor_ref, **kwargs):
38 """
39 Creates an instance of a server.
40 name (Required): The name of the server.
41 image_ref (Required): Reference to the image used to build the server.
42 flavor_ref (Required): The flavor used to build the server.
43 Following optional keyword arguments are accepted:
44 adminPass: Sets the initial root password.
45 key_name: Key name of keypair that was created earlier.
46 meta: A dictionary of values to be used as metadata.
47 personality: A list of dictionaries for files to be injected into
48 the server.
49 security_groups: A list of security group dicts.
50 networks: A list of network dicts with UUID and fixed_ip.
51 user_data: User data for instance.
52 availability_zone: Availability zone in which to launch instance.
53 accessIPv4: The IPv4 access address for the server.
54 accessIPv6: The IPv6 access address for the server.
55 min_count: Count of minimum number of instances to launch.
56 max_count: Count of maximum number of instances to launch.
57 disk_config: Determines if user or admin controls disk configuration.
58 return_reservation_id: Enable/Disable the return of reservation id
59 """
60 post_body = {
61 'name': name,
62 'imageRef': image_ref,
63 'flavorRef': flavor_ref
64 }
65
66 for option in ['personality', 'adminPass', 'key_name',
67 'security_groups', 'networks', 'user_data',
68 'availability_zone', 'accessIPv4', 'accessIPv6',
69 'min_count', 'max_count', ('metadata', 'meta'),
70 ('OS-DCF:diskConfig', 'disk_config'),
71 'return_reservation_id']:
72 if isinstance(option, tuple):
73 post_param = option[0]
74 key = option[1]
75 else:
76 post_param = option
77 key = option
78 value = kwargs.get(key)
79 if value is not None:
80 post_body[post_param] = value
81 post_body = json.dumps({'server': post_body})
82 resp, body = self.post('servers', post_body, self.headers)
83
84 body = json.loads(body)
85 # NOTE(maurosr): this deals with the case of multiple server create
86 # with return reservation id set True
87 if 'reservation_id' in body:
88 return resp, body
89 return resp, body['server']
90
91 def update_server(self, server_id, name=None, meta=None, accessIPv4=None,
92 accessIPv6=None, disk_config=None):
93 """
94 Updates the properties of an existing server.
95 server_id: The id of an existing server.
96 name: The name of the server.
97 personality: A list of files to be injected into the server.
98 accessIPv4: The IPv4 access address for the server.
99 accessIPv6: The IPv6 access address for the server.
100 """
101
102 post_body = {}
103
104 if meta is not None:
105 post_body['metadata'] = meta
106
107 if name is not None:
108 post_body['name'] = name
109
110 if accessIPv4 is not None:
111 post_body['accessIPv4'] = accessIPv4
112
113 if accessIPv6 is not None:
114 post_body['accessIPv6'] = accessIPv6
115
116 if disk_config is not None:
117 post_body['OS-DCF:diskConfig'] = disk_config
118
119 post_body = json.dumps({'server': post_body})
120 resp, body = self.put("servers/%s" % str(server_id),
121 post_body, self.headers)
122 body = json.loads(body)
123 return resp, body['server']
124
125 def get_server(self, server_id):
126 """Returns the details of an existing server."""
127 resp, body = self.get("servers/%s" % str(server_id))
128 body = json.loads(body)
129 return resp, body['server']
130
131 def delete_server(self, server_id):
132 """Deletes the given server."""
133 return self.delete("servers/%s" % str(server_id))
134
135 def list_servers(self, params=None):
136 """Lists all servers for a user."""
137
138 url = 'servers'
139 if params:
140 url += '?%s' % urllib.urlencode(params)
141
142 resp, body = self.get(url)
143 body = json.loads(body)
144 return resp, body
145
146 def list_servers_with_detail(self, params=None):
147 """Lists all servers in detail for a user."""
148
149 url = 'servers/detail'
150 if params:
151 url += '?%s' % urllib.urlencode(params)
152
153 resp, body = self.get(url)
154 body = json.loads(body)
155 return resp, body
156
157 def wait_for_server_status(self, server_id, status):
158 """Waits for a server to reach a given status."""
159 return waiters.wait_for_server_status(self, server_id, status)
160
161 def wait_for_server_termination(self, server_id, ignore_error=False):
162 """Waits for server to reach termination."""
163 start_time = int(time.time())
164 while True:
165 try:
166 resp, body = self.get_server(server_id)
167 except exceptions.NotFound:
168 return
169
170 server_status = body['status']
171 if server_status == 'ERROR' and not ignore_error:
172 raise exceptions.BuildErrorException(server_id=server_id)
173
174 if int(time.time()) - start_time >= self.build_timeout:
175 raise exceptions.TimeoutException
176
177 time.sleep(self.build_interval)
178
179 def list_addresses(self, server_id):
180 """Lists all addresses for a server."""
181 resp, body = self.get("servers/%s/ips" % str(server_id))
182 body = json.loads(body)
183 return resp, body['addresses']
184
185 def list_addresses_by_network(self, server_id, network_id):
186 """Lists all addresses of a specific network type for a server."""
187 resp, body = self.get("servers/%s/ips/%s" %
188 (str(server_id), network_id))
189 body = json.loads(body)
190 return resp, body
191
192 def action(self, server_id, action_name, response_key, **kwargs):
193 post_body = json.dumps({action_name: kwargs})
194 resp, body = self.post('servers/%s/action' % str(server_id),
195 post_body, self.headers)
196 if response_key is not None:
197 body = json.loads(body)[response_key]
198 return resp, body
199
200 def change_password(self, server_id, adminPass):
201 """Changes the root password for the server."""
202 return self.action(server_id, 'changePassword', None,
203 adminPass=adminPass)
204
205 def reboot(self, server_id, reboot_type):
206 """Reboots a server."""
207 return self.action(server_id, 'reboot', None, type=reboot_type)
208
209 def rebuild(self, server_id, image_ref, **kwargs):
210 """Rebuilds a server with a new image."""
211 kwargs['imageRef'] = image_ref
212 if 'disk_config' in kwargs:
213 kwargs['OS-DCF:diskConfig'] = kwargs['disk_config']
214 del kwargs['disk_config']
215 return self.action(server_id, 'rebuild', 'server', **kwargs)
216
217 def resize(self, server_id, flavor_ref, **kwargs):
218 """Changes the flavor of a server."""
219 kwargs['flavorRef'] = flavor_ref
220 if 'disk_config' in kwargs:
221 kwargs['OS-DCF:diskConfig'] = kwargs['disk_config']
222 del kwargs['disk_config']
223 return self.action(server_id, 'resize', None, **kwargs)
224
225 def confirm_resize(self, server_id, **kwargs):
226 """Confirms the flavor change for a server."""
227 return self.action(server_id, 'confirmResize', None, **kwargs)
228
229 def revert_resize(self, server_id, **kwargs):
230 """Reverts a server back to its original flavor."""
231 return self.action(server_id, 'revertResize', None, **kwargs)
232
233 def create_image(self, server_id, name):
234 """Creates an image of the given server."""
235 return self.action(server_id, 'createImage', None, name=name)
236
237 def list_server_metadata(self, server_id):
238 resp, body = self.get("servers/%s/metadata" % str(server_id))
239 body = json.loads(body)
240 return resp, body['metadata']
241
242 def set_server_metadata(self, server_id, meta, no_metadata_field=False):
243 if no_metadata_field:
244 post_body = ""
245 else:
246 post_body = json.dumps({'metadata': meta})
247 resp, body = self.put('servers/%s/metadata' % str(server_id),
248 post_body, self.headers)
249 body = json.loads(body)
250 return resp, body['metadata']
251
252 def update_server_metadata(self, server_id, meta):
253 post_body = json.dumps({'metadata': meta})
254 resp, body = self.post('servers/%s/metadata' % str(server_id),
255 post_body, self.headers)
256 body = json.loads(body)
257 return resp, body['metadata']
258
259 def get_server_metadata_item(self, server_id, key):
260 resp, body = self.get("servers/%s/metadata/%s" % (str(server_id), key))
261 body = json.loads(body)
262 return resp, body['meta']
263
264 def set_server_metadata_item(self, server_id, key, meta):
265 post_body = json.dumps({'meta': meta})
266 resp, body = self.put('servers/%s/metadata/%s' % (str(server_id), key),
267 post_body, self.headers)
268 body = json.loads(body)
269 return resp, body['meta']
270
271 def delete_server_metadata_item(self, server_id, key):
272 resp, body = self.delete("servers/%s/metadata/%s" %
273 (str(server_id), key))
274 return resp, body
275
276 def stop(self, server_id, **kwargs):
277 return self.action(server_id, 'os-stop', None, **kwargs)
278
279 def start(self, server_id, **kwargs):
280 return self.action(server_id, 'os-start', None, **kwargs)
281
282 def attach_volume(self, server_id, volume_id, device='/dev/vdz'):
283 """Attaches a volume to a server instance."""
284 post_body = json.dumps({
285 'volumeAttachment': {
286 'volumeId': volume_id,
287 'device': device,
288 }
289 })
290 resp, body = self.post('servers/%s/os-volume_attachments' % server_id,
291 post_body, self.headers)
292 return resp, body
293
294 def detach_volume(self, server_id, volume_id):
295 """Detaches a volume from a server instance."""
296 resp, body = self.delete('servers/%s/os-volume_attachments/%s' %
297 (server_id, volume_id))
298 return resp, body
299
300 def add_security_group(self, server_id, name):
301 """Adds a security group to the server."""
302 return self.action(server_id, 'addSecurityGroup', None, name=name)
303
304 def remove_security_group(self, server_id, name):
305 """Removes a security group from the server."""
306 return self.action(server_id, 'removeSecurityGroup', None, name=name)
307
308 def live_migrate_server(self, server_id, dest_host, use_block_migration):
309 """This should be called with administrator privileges ."""
310
311 migrate_params = {
312 "disk_over_commit": False,
313 "block_migration": use_block_migration,
314 "host": dest_host
315 }
316
317 req_body = json.dumps({'os-migrateLive': migrate_params})
318
319 resp, body = self.post("servers/%s/action" % str(server_id),
320 req_body, self.headers)
321 return resp, body
322
323 def migrate_server(self, server_id, **kwargs):
324 """Migrates a server to a new host."""
325 return self.action(server_id, 'migrate', None, **kwargs)
326
327 def lock_server(self, server_id, **kwargs):
328 """Locks the given server."""
329 return self.action(server_id, 'lock', None, **kwargs)
330
331 def unlock_server(self, server_id, **kwargs):
332 """UNlocks the given server."""
333 return self.action(server_id, 'unlock', None, **kwargs)
334
335 def suspend_server(self, server_id, **kwargs):
336 """Suspends the provded server."""
337 return self.action(server_id, 'suspend', None, **kwargs)
338
339 def resume_server(self, server_id, **kwargs):
340 """Un-suspends the provded server."""
341 return self.action(server_id, 'resume', None, **kwargs)
342
343 def pause_server(self, server_id, **kwargs):
344 """Pauses the provded server."""
345 return self.action(server_id, 'pause', None, **kwargs)
346
347 def unpause_server(self, server_id, **kwargs):
348 """Un-pauses the provded server."""
349 return self.action(server_id, 'unpause', None, **kwargs)
350
351 def reset_state(self, server_id, state='error'):
352 """Resets the state of a server to active/error."""
353 return self.action(server_id, 'os-resetState', None, state=state)
354
355 def get_console_output(self, server_id, length):
356 return self.action(server_id, 'os-getConsoleOutput', 'output',
357 length=length)
358
359 def list_virtual_interfaces(self, server_id):
360 """
361 List the virtual interfaces used in an instance.
362 """
363 resp, body = self.get('/'.join(['servers', server_id,
364 'os-virtual-interfaces']))
365 return resp, json.loads(body)
366
367 def rescue_server(self, server_id, adminPass=None):
368 """Rescue the provided server."""
369 return self.action(server_id, 'rescue', None, adminPass=adminPass)
370
371 def unrescue_server(self, server_id):
372 """Unrescue the provided server."""
373 return self.action(server_id, 'unrescue', None)
374
375 def get_server_diagnostics(self, server_id):
376 """Get the usage data for a server."""
377 resp, body = self.get("servers/%s/diagnostics" % str(server_id))
378 return resp, json.loads(body)
379
380 def list_instance_actions(self, server_id):
381 """List the provided server action."""
382 resp, body = self.get("servers/%s/os-instance-actions" %
383 str(server_id))
384 body = json.loads(body)
385 return resp, body['instanceActions']
386
387 def get_instance_action(self, server_id, request_id):
388 """Returns the action details of the provided server."""
389 resp, body = self.get("servers/%s/os-instance-actions/%s" %
390 (str(server_id), str(request_id)))
391 body = json.loads(body)
392 return resp, body['instanceAction']