Evacuate each instance from one host to another
Added a new extension that adds the ability for admins to evacuate an entire host to another host. This internally uses the existing server.evacuate api. The target host is optional so that a free host will be chosen by the scheduler in the api. Implements: blueprint evacuate-host Change-Id: I2352836d01952e281e15edb9bdd1b912106516d6
This commit is contained in:
parent
f67c5e0cf9
commit
0d678ed4bb
@ -1702,3 +1702,15 @@ class FakeHTTPClient(base_client.HTTPClient):
|
|||||||
"action": "create",
|
"action": "create",
|
||||||
"message": None,
|
"message": None,
|
||||||
"project_id": "04019601fe3648c0abd4f4abfb9e6106"}})
|
"project_id": "04019601fe3648c0abd4f4abfb9e6106"}})
|
||||||
|
|
||||||
|
def post_servers_uuid1_action(self, **kw):
|
||||||
|
return 202, {}, {}
|
||||||
|
|
||||||
|
def post_servers_uuid2_action(self, **kw):
|
||||||
|
return 202, {}, {}
|
||||||
|
|
||||||
|
def post_servers_uuid3_action(self, **kw):
|
||||||
|
return 202, {}, {}
|
||||||
|
|
||||||
|
def post_servers_uuid4_action(self, **kw):
|
||||||
|
return 202, {}, {}
|
||||||
|
@ -942,6 +942,55 @@ class ShellTest(utils.TestCase):
|
|||||||
self.assert_called(
|
self.assert_called(
|
||||||
'POST', '/os-hosts/sample-host/action', {'reboot': None})
|
'POST', '/os-hosts/sample-host/action', {'reboot': None})
|
||||||
|
|
||||||
|
def test_host_evacuate(self):
|
||||||
|
self.run_command('host-evacuate hyper --target target_hyper')
|
||||||
|
self.assert_called('GET', '/os-hypervisors/hyper/servers', pos=0)
|
||||||
|
self.assert_called('POST', '/servers/uuid1/action',
|
||||||
|
{'evacuate': {'host': 'target_hyper',
|
||||||
|
'onSharedStorage': False}}, pos=1)
|
||||||
|
self.assert_called('POST', '/servers/uuid2/action',
|
||||||
|
{'evacuate': {'host': 'target_hyper',
|
||||||
|
'onSharedStorage': False}}, pos=2)
|
||||||
|
self.assert_called('POST', '/servers/uuid3/action',
|
||||||
|
{'evacuate': {'host': 'target_hyper',
|
||||||
|
'onSharedStorage': False}}, pos=3)
|
||||||
|
self.assert_called('POST', '/servers/uuid4/action',
|
||||||
|
{'evacuate': {'host': 'target_hyper',
|
||||||
|
'onSharedStorage': False}}, pos=4)
|
||||||
|
|
||||||
|
def test_host_evacuate_with_shared_storage(self):
|
||||||
|
self.run_command(
|
||||||
|
'host-evacuate --on-shared-storage hyper --target target_hyper')
|
||||||
|
self.assert_called('GET', '/os-hypervisors/hyper/servers', pos=0)
|
||||||
|
self.assert_called('POST', '/servers/uuid1/action',
|
||||||
|
{'evacuate': {'host': 'target_hyper',
|
||||||
|
'onSharedStorage': True}}, pos=1)
|
||||||
|
self.assert_called('POST', '/servers/uuid2/action',
|
||||||
|
{'evacuate': {'host': 'target_hyper',
|
||||||
|
'onSharedStorage': True}}, pos=2)
|
||||||
|
self.assert_called('POST', '/servers/uuid3/action',
|
||||||
|
{'evacuate': {'host': 'target_hyper',
|
||||||
|
'onSharedStorage': True}}, pos=3)
|
||||||
|
self.assert_called('POST', '/servers/uuid4/action',
|
||||||
|
{'evacuate': {'host': 'target_hyper',
|
||||||
|
'onSharedStorage': True}}, pos=4)
|
||||||
|
|
||||||
|
def test_host_evacuate_with_no_target_host(self):
|
||||||
|
self.run_command('host-evacuate --on-shared-storage hyper')
|
||||||
|
self.assert_called('GET', '/os-hypervisors/hyper/servers', pos=0)
|
||||||
|
self.assert_called('POST', '/servers/uuid1/action',
|
||||||
|
{'evacuate': {'host': None,
|
||||||
|
'onSharedStorage': True}}, pos=1)
|
||||||
|
self.assert_called('POST', '/servers/uuid2/action',
|
||||||
|
{'evacuate': {'host': None,
|
||||||
|
'onSharedStorage': True}}, pos=2)
|
||||||
|
self.assert_called('POST', '/servers/uuid3/action',
|
||||||
|
{'evacuate': {'host': None,
|
||||||
|
'onSharedStorage': True}}, pos=3)
|
||||||
|
self.assert_called('POST', '/servers/uuid4/action',
|
||||||
|
{'evacuate': {'host': None,
|
||||||
|
'onSharedStorage': True}}, pos=4)
|
||||||
|
|
||||||
def test_coverage_start(self):
|
def test_coverage_start(self):
|
||||||
self.run_command('coverage-start')
|
self.run_command('coverage-start')
|
||||||
self.assert_called('POST', '/os-coverage/action')
|
self.assert_called('POST', '/os-coverage/action')
|
||||||
|
59
novaclient/v1_1/contrib/host_evacuate.py
Normal file
59
novaclient/v1_1/contrib/host_evacuate.py
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
# Copyright 2013 Rackspace Hosting
|
||||||
|
# All Rights Reserved.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
from novaclient import base
|
||||||
|
from novaclient import utils
|
||||||
|
|
||||||
|
|
||||||
|
class EvacuateHostResponse(base.Resource):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def _server_evacuate(cs, server, args):
|
||||||
|
success = True
|
||||||
|
error_message = ""
|
||||||
|
try:
|
||||||
|
cs.servers.evacuate(server['uuid'], args.target_host,
|
||||||
|
args.on_shared_storage)
|
||||||
|
except Exception as e:
|
||||||
|
success = False
|
||||||
|
error_message = "Error while evacuating instance: %s" % e
|
||||||
|
return EvacuateHostResponse(base.Manager,
|
||||||
|
{"server_uuid": server['uuid'],
|
||||||
|
"evacuate_accepted": success,
|
||||||
|
"error_message": error_message})
|
||||||
|
|
||||||
|
|
||||||
|
@utils.arg('host', metavar='<host>', help='Name of host.')
|
||||||
|
@utils.arg('--target_host',
|
||||||
|
metavar='<target_host>',
|
||||||
|
default=None,
|
||||||
|
help='Name of target host.')
|
||||||
|
@utils.arg('--on-shared-storage',
|
||||||
|
dest='on_shared_storage',
|
||||||
|
action="store_true",
|
||||||
|
default=False,
|
||||||
|
help='Specifies whether all instances files are on shared storage')
|
||||||
|
def do_host_evacuate(cs, args):
|
||||||
|
"""Evacuate all instances from failed host to specified one."""
|
||||||
|
hypervisors = cs.hypervisors.search(args.host, servers=True)
|
||||||
|
response = []
|
||||||
|
for hyper in hypervisors:
|
||||||
|
if hasattr(hyper, 'servers'):
|
||||||
|
for server in hyper.servers:
|
||||||
|
response.append(_server_evacuate(cs, server, args))
|
||||||
|
|
||||||
|
utils.print_list(response,
|
||||||
|
["Server UUID", "Evacuate Accepted", "Error Message"])
|
Loading…
x
Reference in New Issue
Block a user