Merge "compute: Remove deprecated 'server migrate --live' option"

This commit is contained in:
Zuul 2021-03-04 21:29:55 +00:00 committed by Gerrit Code Review
commit a507fb50f8
3 changed files with 67 additions and 219 deletions

View File

@ -2473,27 +2473,7 @@ revert to release the new server and restart the old one.""")
'validated by the scheduler' 'validated by the scheduler'
), ),
) )
# The --live and --host options are mutually exclusive ways of asking parser.add_argument(
# for a target host during a live migration.
host_group = parser.add_mutually_exclusive_group()
# TODO(mriedem): Remove --live in the next major version bump after
# the Train release.
host_group.add_argument(
'--live',
metavar='<hostname>',
help=_(
'**Deprecated** This option is problematic in that it '
'requires a host and prior to compute API version 2.30, '
'specifying a host during live migration will bypass '
'validation by the scheduler which could result in '
'failures to actually migrate the server to the specified '
'host or over-subscribe the host. Use the '
'``--live-migration`` option instead. If both this option '
'and ``--live-migration`` are used, ``--live-migration`` '
'takes priority.'
),
)
host_group.add_argument(
'--host', '--host',
metavar='<hostname>', metavar='<hostname>',
help=_( help=_(
@ -2551,15 +2531,6 @@ revert to release the new server and restart the old one.""")
) )
return parser return parser
def _log_warning_for_live(self, parsed_args):
if parsed_args.live:
# NOTE(mriedem): The --live option requires a host and if
# --os-compute-api-version is less than 2.30 it will forcefully
# bypass the scheduler which is dangerous.
self.log.warning(_(
'The --live option has been deprecated. Please use the '
'--live-migration option instead.'))
def take_action(self, parsed_args): def take_action(self, parsed_args):
def _show_progress(progress): def _show_progress(progress):
@ -2573,11 +2544,8 @@ revert to release the new server and restart the old one.""")
compute_client.servers, compute_client.servers,
parsed_args.server, parsed_args.server,
) )
# Check for live migration.
if parsed_args.live or parsed_args.live_migration:
# Always log a warning if --live is used.
self._log_warning_for_live(parsed_args)
if parsed_args.live_migration:
kwargs = {} kwargs = {}
block_migration = parsed_args.block_migration block_migration = parsed_args.block_migration
@ -2592,28 +2560,23 @@ revert to release the new server and restart the old one.""")
kwargs['block_migration'] = block_migration kwargs['block_migration'] = block_migration
# Prefer --live-migration over --live if both are specified. # Technically we could pass a non-None host with
if parsed_args.live_migration: # --os-compute-api-version < 2.30 but that is the same thing
# Technically we could pass a non-None host with # as the --live option bypassing the scheduler which we don't
# --os-compute-api-version < 2.30 but that is the same thing # want to support, so if the user is using --live-migration
# as the --live option bypassing the scheduler which we don't # and --host, we want to enforce that they are using version
# want to support, so if the user is using --live-migration # 2.30 or greater.
# and --host, we want to enforce that they are using version if (
# 2.30 or greater. parsed_args.host and
if ( compute_client.api_version < api_versions.APIVersion('2.30')
parsed_args.host and ):
compute_client.api_version < raise exceptions.CommandError(
api_versions.APIVersion('2.30') '--os-compute-api-version 2.30 or greater is required '
): 'when using --host'
raise exceptions.CommandError( )
'--os-compute-api-version 2.30 or greater is required '
'when using --host'
)
# The host parameter is required in the API even if None. # The host parameter is required in the API even if None.
kwargs['host'] = parsed_args.host kwargs['host'] = parsed_args.host
else:
kwargs['host'] = parsed_args.live
if compute_client.api_version < api_versions.APIVersion('2.25'): if compute_client.api_version < api_versions.APIVersion('2.25'):
kwargs['disk_over_commit'] = parsed_args.disk_overcommit kwargs['disk_over_commit'] = parsed_args.disk_overcommit
@ -2628,7 +2591,7 @@ revert to release the new server and restart the old one.""")
self.log.warning(msg) self.log.warning(msg)
server.live_migrate(**kwargs) server.live_migrate(**kwargs)
else: else: # cold migration
if parsed_args.block_migration or parsed_args.disk_overcommit: if parsed_args.block_migration or parsed_args.disk_overcommit:
raise exceptions.CommandError( raise exceptions.CommandError(
"--live-migration must be specified if " "--live-migration must be specified if "

View File

@ -4523,7 +4523,7 @@ class TestServerMigrate(TestServer):
self.server.id, self.server.id,
] ]
verifylist = [ verifylist = [
('live', None), ('live_migration', False),
('block_migration', None), ('block_migration', None),
('disk_overcommit', False), ('disk_overcommit', False),
('wait', False), ('wait', False),
@ -4544,7 +4544,6 @@ class TestServerMigrate(TestServer):
'--host', 'fakehost', self.server.id, '--host', 'fakehost', self.server.id,
] ]
verifylist = [ verifylist = [
('live', None),
('live_migration', False), ('live_migration', False),
('host', 'fakehost'), ('host', 'fakehost'),
('block_migration', None), ('block_migration', None),
@ -4568,7 +4567,7 @@ class TestServerMigrate(TestServer):
'--block-migration', self.server.id, '--block-migration', self.server.id,
] ]
verifylist = [ verifylist = [
('live', None), ('live_migration', False),
('block_migration', True), ('block_migration', True),
('disk_overcommit', False), ('disk_overcommit', False),
('wait', False), ('wait', False),
@ -4587,7 +4586,7 @@ class TestServerMigrate(TestServer):
'--disk-overcommit', self.server.id, '--disk-overcommit', self.server.id,
] ]
verifylist = [ verifylist = [
('live', None), ('live_migration', False),
('block_migration', None), ('block_migration', None),
('disk_overcommit', True), ('disk_overcommit', True),
('wait', False), ('wait', False),
@ -4608,7 +4607,6 @@ class TestServerMigrate(TestServer):
'--host', 'fakehost', self.server.id, '--host', 'fakehost', self.server.id,
] ]
verifylist = [ verifylist = [
('live', None),
('live_migration', False), ('live_migration', False),
('host', 'fakehost'), ('host', 'fakehost'),
('block_migration', None), ('block_migration', None),
@ -4630,70 +4628,11 @@ class TestServerMigrate(TestServer):
self.assertNotCalled(self.servers_mock.migrate) self.assertNotCalled(self.servers_mock.migrate)
def test_server_live_migrate(self): def test_server_live_migrate(self):
arglist = [
'--live', 'fakehost', self.server.id,
]
verifylist = [
('live', 'fakehost'),
('live_migration', False),
('host', None),
('block_migration', None),
('disk_overcommit', False),
('wait', False),
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
self.app.client_manager.compute.api_version = \
api_versions.APIVersion('2.24')
with mock.patch.object(self.cmd.log, 'warning') as mock_warning:
result = self.cmd.take_action(parsed_args)
self.servers_mock.get.assert_called_with(self.server.id)
self.server.live_migrate.assert_called_with(block_migration=False,
disk_over_commit=False,
host='fakehost')
self.assertNotCalled(self.servers_mock.migrate)
self.assertIsNone(result)
# A warning should have been logged for using --live.
mock_warning.assert_called_once()
self.assertIn('The --live option has been deprecated.',
str(mock_warning.call_args[0][0]))
def test_server_live_migrate_host_pre_2_30(self):
# Tests that the --host option is not supported for --live-migration
# before microversion 2.30 (the test defaults to 2.1).
arglist = [
'--live-migration', '--host', 'fakehost', self.server.id,
]
verifylist = [
('live', None),
('live_migration', True),
('host', 'fakehost'),
('block_migration', None),
('disk_overcommit', False),
('wait', False),
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
ex = self.assertRaises(exceptions.CommandError, self.cmd.take_action,
parsed_args)
# Make sure it's the error we expect.
self.assertIn('--os-compute-api-version 2.30 or greater is required '
'when using --host', str(ex))
self.servers_mock.get.assert_called_with(self.server.id)
self.assertNotCalled(self.servers_mock.live_migrate)
self.assertNotCalled(self.servers_mock.migrate)
def test_server_live_migrate_no_host(self):
# Tests the --live-migration option without --host or --live. # Tests the --live-migration option without --host or --live.
arglist = [ arglist = [
'--live-migration', self.server.id, '--live-migration', self.server.id,
] ]
verifylist = [ verifylist = [
('live', None),
('live_migration', True), ('live_migration', True),
('host', None), ('host', None),
('block_migration', None), ('block_migration', None),
@ -4702,26 +4641,22 @@ class TestServerMigrate(TestServer):
] ]
parsed_args = self.check_parser(self.cmd, arglist, verifylist) parsed_args = self.check_parser(self.cmd, arglist, verifylist)
with mock.patch.object(self.cmd.log, 'warning') as mock_warning: result = self.cmd.take_action(parsed_args)
result = self.cmd.take_action(parsed_args)
self.servers_mock.get.assert_called_with(self.server.id) self.servers_mock.get.assert_called_with(self.server.id)
self.server.live_migrate.assert_called_with(block_migration=False, self.server.live_migrate.assert_called_with(
disk_over_commit=False, block_migration=False,
host=None) disk_over_commit=False,
host=None)
self.assertNotCalled(self.servers_mock.migrate) self.assertNotCalled(self.servers_mock.migrate)
self.assertIsNone(result) self.assertIsNone(result)
# Since --live wasn't used a warning shouldn't have been logged.
mock_warning.assert_not_called()
def test_server_live_migrate_with_host(self): def test_server_live_migrate_with_host(self):
# Tests the --live-migration option with --host but no --live.
# This requires --os-compute-api-version >= 2.30 so the test uses 2.30. # This requires --os-compute-api-version >= 2.30 so the test uses 2.30.
arglist = [ arglist = [
'--live-migration', '--host', 'fakehost', self.server.id, '--live-migration', '--host', 'fakehost', self.server.id,
] ]
verifylist = [ verifylist = [
('live', None),
('live_migration', True), ('live_migration', True),
('host', 'fakehost'), ('host', 'fakehost'),
('block_migration', None), ('block_migration', None),
@ -4743,53 +4678,42 @@ class TestServerMigrate(TestServer):
self.assertNotCalled(self.servers_mock.migrate) self.assertNotCalled(self.servers_mock.migrate)
self.assertIsNone(result) self.assertIsNone(result)
def test_server_live_migrate_without_host_override_live(self): def test_server_live_migrate_with_host_pre_v230(self):
# Tests the --live-migration option without --host and with --live. # Tests that the --host option is not supported for --live-migration
# The --live-migration option will take precedence and a warning is # before microversion 2.30 (the test defaults to 2.1).
# logged for using --live.
arglist = [ arglist = [
'--live', 'fakehost', '--live-migration', self.server.id, '--live-migration', '--host', 'fakehost', self.server.id,
] ]
verifylist = [ verifylist = [
('live', 'fakehost'),
('live_migration', True), ('live_migration', True),
('host', None), ('host', 'fakehost'),
('block_migration', None), ('block_migration', None),
('disk_overcommit', False), ('disk_overcommit', False),
('wait', False), ('wait', False),
] ]
parsed_args = self.check_parser(self.cmd, arglist, verifylist) parsed_args = self.check_parser(self.cmd, arglist, verifylist)
with mock.patch.object(self.cmd.log, 'warning') as mock_warning: ex = self.assertRaises(
result = self.cmd.take_action(parsed_args) exceptions.CommandError, self.cmd.take_action,
parsed_args)
# Make sure it's the error we expect.
self.assertIn(
'--os-compute-api-version 2.30 or greater is required '
'when using --host', str(ex))
self.servers_mock.get.assert_called_with(self.server.id) self.servers_mock.get.assert_called_with(self.server.id)
self.server.live_migrate.assert_called_with(block_migration=False, self.assertNotCalled(self.servers_mock.live_migrate)
disk_over_commit=False,
host=None)
self.assertNotCalled(self.servers_mock.migrate) self.assertNotCalled(self.servers_mock.migrate)
self.assertIsNone(result)
# A warning should have been logged for using --live.
mock_warning.assert_called_once()
self.assertIn('The --live option has been deprecated.',
str(mock_warning.call_args[0][0]))
def test_server_live_migrate_live_and_host_mutex(self):
# Tests specifying both the --live and --host options which are in a
# mutex group so argparse should fail.
arglist = [
'--live', 'fakehost', '--host', 'fakehost', self.server.id,
]
self.assertRaises(
utils.ParserException,
self.check_parser, self.cmd, arglist, verify_args=[])
def test_server_block_live_migrate(self): def test_server_block_live_migrate(self):
arglist = [ arglist = [
'--live', 'fakehost', '--block-migration', self.server.id, '--live-migration',
'--block-migration',
self.server.id,
] ]
verifylist = [ verifylist = [
('live', 'fakehost'), ('live_migration', True),
('block_migration', True), ('block_migration', True),
('disk_overcommit', False), ('disk_overcommit', False),
('wait', False), ('wait', False),
@ -4802,18 +4726,21 @@ class TestServerMigrate(TestServer):
result = self.cmd.take_action(parsed_args) result = self.cmd.take_action(parsed_args)
self.servers_mock.get.assert_called_with(self.server.id) self.servers_mock.get.assert_called_with(self.server.id)
self.server.live_migrate.assert_called_with(block_migration=True, self.server.live_migrate.assert_called_with(
disk_over_commit=False, block_migration=True,
host='fakehost') disk_over_commit=False,
host=None)
self.assertNotCalled(self.servers_mock.migrate) self.assertNotCalled(self.servers_mock.migrate)
self.assertIsNone(result) self.assertIsNone(result)
def test_server_live_migrate_with_disk_overcommit(self): def test_server_live_migrate_with_disk_overcommit(self):
arglist = [ arglist = [
'--live', 'fakehost', '--disk-overcommit', self.server.id, '--live-migration',
'--disk-overcommit',
self.server.id,
] ]
verifylist = [ verifylist = [
('live', 'fakehost'), ('live_migration', True),
('block_migration', None), ('block_migration', None),
('disk_overcommit', True), ('disk_overcommit', True),
('wait', False), ('wait', False),
@ -4826,9 +4753,10 @@ class TestServerMigrate(TestServer):
result = self.cmd.take_action(parsed_args) result = self.cmd.take_action(parsed_args)
self.servers_mock.get.assert_called_with(self.server.id) self.servers_mock.get.assert_called_with(self.server.id)
self.server.live_migrate.assert_called_with(block_migration=False, self.server.live_migrate.assert_called_with(
disk_over_commit=True, block_migration=False,
host='fakehost') disk_over_commit=True,
host=None)
self.assertNotCalled(self.servers_mock.migrate) self.assertNotCalled(self.servers_mock.migrate)
self.assertIsNone(result) self.assertIsNone(result)
@ -4839,7 +4767,6 @@ class TestServerMigrate(TestServer):
self.server.id, self.server.id,
] ]
verifylist = [ verifylist = [
('live', None),
('live_migration', True), ('live_migration', True),
('block_migration', None), ('block_migration', None),
('disk_overcommit', True), ('disk_overcommit', True),
@ -4866,63 +4793,13 @@ class TestServerMigrate(TestServer):
'The --disk-overcommit and --no-disk-overcommit options ', 'The --disk-overcommit and --no-disk-overcommit options ',
str(mock_warning.call_args[0][0])) str(mock_warning.call_args[0][0]))
def test_server_live_migrate_with_false_value_options(self):
arglist = [
'--live', 'fakehost', '--no-disk-overcommit',
'--shared-migration', self.server.id,
]
verifylist = [
('live', 'fakehost'),
('block_migration', False),
('disk_overcommit', False),
('wait', False),
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
self.app.client_manager.compute.api_version = \
api_versions.APIVersion('2.24')
result = self.cmd.take_action(parsed_args)
self.servers_mock.get.assert_called_with(self.server.id)
self.server.live_migrate.assert_called_with(block_migration=False,
disk_over_commit=False,
host='fakehost')
self.assertNotCalled(self.servers_mock.migrate)
self.assertIsNone(result)
def test_server_live_migrate_225(self):
arglist = [
'--live', 'fakehost', self.server.id,
]
verifylist = [
('live', 'fakehost'),
('block_migration', None),
('wait', False),
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
self.app.client_manager.compute.api_version = \
api_versions.APIVersion('2.25')
result = self.cmd.take_action(parsed_args)
self.servers_mock.get.assert_called_with(self.server.id)
# No disk_overcommit and block_migration defaults to auto with
# microversion >= 2.25
self.server.live_migrate.assert_called_with(
block_migration='auto',
host='fakehost')
self.assertNotCalled(self.servers_mock.migrate)
self.assertIsNone(result)
@mock.patch.object(common_utils, 'wait_for_status', return_value=True) @mock.patch.object(common_utils, 'wait_for_status', return_value=True)
def test_server_migrate_with_wait(self, mock_wait_for_status): def test_server_migrate_with_wait(self, mock_wait_for_status):
arglist = [ arglist = [
'--wait', self.server.id, '--wait', self.server.id,
] ]
verifylist = [ verifylist = [
('live', None), ('live_migration', False),
('block_migration', None), ('block_migration', None),
('disk_overcommit', False), ('disk_overcommit', False),
('wait', True), ('wait', True),
@ -4942,7 +4819,7 @@ class TestServerMigrate(TestServer):
'--wait', self.server.id, '--wait', self.server.id,
] ]
verifylist = [ verifylist = [
('live', None), ('live_migration', False),
('block_migration', None), ('block_migration', None),
('disk_overcommit', False), ('disk_overcommit', False),
('wait', True), ('wait', True),

View File

@ -0,0 +1,8 @@
---
upgrade:
- |
The deprecated ``--live`` option of the ``server migrate`` command has
been removed. This was problematic as it required a host argument and
would result in a forced live migration to a host, bypassing the
scheduler. It has been replaced by a ``--live-migration`` option and
optional ``--host`` option.