Fix openstack server list --deleted --marker option

This patch removes using the "name" option for a marker when
--deleted is also used. The find_resource() function
that is being called does not correctly handle using the marker
as the "name" in the search when also using deleted=True.
One simple way to fix this is force the marker to only be an ID
when --deleted is used. This is how the nova client works.

Using the --deleted option is available to users with the admin
role by default. If you're an admin listing --deleted servers
with a marker by name, find_resource() is going to fail to find
it since it doesn't apply the --deleted filter to find_resource().

The find_resource() function is trying to find the marker server
by name if it's not found by id, and to find it by name it's
listing servers with the given marker as the name, but not
applying the --deleted filter so it doesn't get back any results.

In the story it was suggested modifying find_resource to include
the deleted query param when it's specified on the command line but
that didn't work because it still results in something like this:

http://192.168.1.123/compute/v2.1/servers?deleted=True&name=4cecd49f-bc25-4a7e-826e-4aea6f9267d9

It seems like there are bugs in find_resource().

Restricting the marker to be the server ID when listing deleted servers
is probably OK since if you're using --deleted you're an admin and you could
be listing across all projects and if you're filtering by a server across all
projects anyway (not that you have to, I'm just saying if you are), or even
showing a server in another project, you have to do it by id rather than name
because find_resource() won't find the server in another project by name, only ID.

story: 2006761
Task: 37258

Change-Id: Ib878982b1d469212ca3483dcfaf407a8e1d2b417
This commit is contained in:
KeithMnemonic 2019-10-24 14:39:50 -04:00 committed by Keith Berger
parent 5b3a827a1f
commit f5384ae16a
3 changed files with 63 additions and 3 deletions

View File

@ -1242,7 +1242,8 @@ class ListServer(command.Lister):
default=None, default=None,
help=_('The last server of the previous page. Display ' help=_('The last server of the previous page. Display '
'list of servers after marker. Display all servers if not ' 'list of servers after marker. Display all servers if not '
'specified. (name or ID)') 'specified. When used with ``--deleted``, the marker must '
'be an ID, otherwise a name or ID can be used.'),
) )
parser.add_argument( parser.add_argument(
'--limit', '--limit',
@ -1450,9 +1451,17 @@ class ListServer(command.Lister):
mixed_case_fields = [] mixed_case_fields = []
marker_id = None marker_id = None
if parsed_args.marker: if parsed_args.marker:
marker_id = utils.find_resource(compute_client.servers, # Check if both "--marker" and "--deleted" are used.
parsed_args.marker).id # In that scenario a lookup is not needed as the marker
# needs to be an ID, because find_resource does not
# handle deleted resources
if parsed_args.deleted:
marker_id = parsed_args.marker
else:
marker_id = utils.find_resource(compute_client.servers,
parsed_args.marker).id
data = compute_client.servers.list(search_opts=search_opts, data = compute_client.servers.list(search_opts=search_opts,
marker=marker_id, marker=marker_id,

View File

@ -63,6 +63,49 @@ class ServerTests(common.ComputeTestCase):
self.assertNotIn(name1, col_name) self.assertNotIn(name1, col_name)
self.assertIn(name2, col_name) self.assertIn(name2, col_name)
def test_server_list_with_marker_and_deleted(self):
"""Test server list with deleted and marker"""
cmd_output = self.server_create(cleanup=False)
name1 = cmd_output['name']
cmd_output = self.server_create(cleanup=False)
name2 = cmd_output['name']
id2 = cmd_output['id']
self.wait_for_status(name1, "ACTIVE")
self.wait_for_status(name2, "ACTIVE")
# Test list --marker with ID
cmd_output = json.loads(self.openstack(
'server list -f json --marker ' + id2
))
col_name = [x["Name"] for x in cmd_output]
self.assertIn(name1, col_name)
# Test list --marker with Name
cmd_output = json.loads(self.openstack(
'server list -f json --marker ' + name2
))
col_name = [x["Name"] for x in cmd_output]
self.assertIn(name1, col_name)
self.openstack('server delete --wait ' + name1)
self.openstack('server delete --wait ' + name2)
# Test list --deleted --marker with ID
cmd_output = json.loads(self.openstack(
'server list -f json --deleted --marker ' + id2
))
col_name = [x["Name"] for x in cmd_output]
self.assertIn(name1, col_name)
# Test list --deleted --marker with Name
try:
cmd_output = json.loads(self.openstack(
'server list -f json --deleted --marker ' + name2
))
except exceptions.CommandFailed as e:
self.assertIn('marker [%s] not found (HTTP 400)' % (name2),
e.stderr.decode('utf-8'))
def test_server_list_with_changes_before(self): def test_server_list_with_changes_before(self):
"""Test server list. """Test server list.

View File

@ -0,0 +1,8 @@
---
fixes:
- |
Fixes the "No server with a name or ID of 'id' exists" error when running
``server list --deleted --marker``. The fix removes using a name for
the marker when both ``--deleted`` and ``--marker`` are used. In
this scenario an ID must be supplied for the marker.
[Story `2006761 <https://storyboard.openstack.org/#!/story/2006761>`_]