Make _console() public

Heat wants to use 'novaclient.v2.servers.
ServerManager._console' method to simplify the
implementation of retrieving server console.
It's better to change the method to public for
public use.
The patch changes:
1. add public method named get_console_url() for class
   'novaclient.v2.servers.Server'
2. rename _console() to get_console_url() for class
   'novaclient.v2.servers.ServerManager'

Change-Id: I3d1485468d1d0a79d4002981ebe05b6cdf2332e7
Closes-Bug: #1651677
This commit is contained in:
huangtianhua 2016-12-22 10:31:29 +08:00
parent ae7b2c1edb
commit 2512a28dc1
4 changed files with 180 additions and 42 deletions

View File

@ -24,6 +24,14 @@ class UnsupportedVersion(Exception):
pass pass
class UnsupportedConsoleType(Exception):
"""Indicates that the user is trying to use an unsupported
console type when retrieving console urls of servers.
"""
def __init__(self, console_type):
self.message = 'Unsupported console_type "%s"' % console_type
class UnsupportedAttribute(AttributeError): class UnsupportedAttribute(AttributeError):
"""Indicates that the user is trying to transmit the argument to a method, """Indicates that the user is trying to transmit the argument to a method,
which is not supported by selected version. which is not supported by selected version.

View File

@ -794,44 +794,59 @@ class ServersTest(utils.FixturedTestCase):
def test_get_vnc_console(self): def test_get_vnc_console(self):
s = self.cs.servers.get(1234) s = self.cs.servers.get(1234)
vc = s.get_vnc_console('fake') vc = s.get_vnc_console('novnc')
self.assert_request_id(vc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(vc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/action') self.assert_called('POST', '/servers/1234/action')
vc = self.cs.servers.get_vnc_console(s, 'fake') vc = self.cs.servers.get_vnc_console(s, 'novnc')
self.assert_request_id(vc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(vc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/action') self.assert_called('POST', '/servers/1234/action')
def test_get_spice_console(self): def test_get_spice_console(self):
s = self.cs.servers.get(1234) s = self.cs.servers.get(1234)
sc = s.get_spice_console('fake') sc = s.get_spice_console('spice-html5')
self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/action') self.assert_called('POST', '/servers/1234/action')
sc = self.cs.servers.get_spice_console(s, 'fake') sc = self.cs.servers.get_spice_console(s, 'spice-html5')
self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/action') self.assert_called('POST', '/servers/1234/action')
def test_get_serial_console(self): def test_get_serial_console(self):
s = self.cs.servers.get(1234) s = self.cs.servers.get(1234)
sc = s.get_serial_console('fake') sc = s.get_serial_console('serial')
self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/action') self.assert_called('POST', '/servers/1234/action')
sc = self.cs.servers.get_serial_console(s, 'fake') sc = self.cs.servers.get_serial_console(s, 'serial')
self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/action') self.assert_called('POST', '/servers/1234/action')
def test_get_rdp_console(self): def test_get_rdp_console(self):
s = self.cs.servers.get(1234) s = self.cs.servers.get(1234)
rc = s.get_rdp_console('fake') rc = s.get_rdp_console('rdp-html5')
self.assert_request_id(rc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(rc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/action') self.assert_called('POST', '/servers/1234/action')
rc = self.cs.servers.get_rdp_console(s, 'fake') rc = self.cs.servers.get_rdp_console(s, 'rdp-html5')
self.assert_request_id(rc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(rc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/action') self.assert_called('POST', '/servers/1234/action')
def test_get_console_url(self):
s = self.cs.servers.get(1234)
rc = s.get_console_url('novnc')
self.assert_request_id(rc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/action')
rc = self.cs.servers.get_console_url(s, 'novnc')
self.assert_request_id(rc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/action')
# test the case with invalid console type
self.assertRaises(exceptions.UnsupportedConsoleType,
s.get_console_url,
'invalid')
def test_create_image(self): def test_create_image(self):
s = self.cs.servers.get(1234) s = self.cs.servers.get(1234)
im = s.create_image('123') im = s.create_image('123')
@ -1001,44 +1016,83 @@ class ServersV26Test(ServersTest):
def test_get_vnc_console(self): def test_get_vnc_console(self):
s = self.cs.servers.get(1234) s = self.cs.servers.get(1234)
vc = s.get_vnc_console('fake') vc = s.get_vnc_console('novnc')
self.assert_request_id(vc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(vc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles') self.assert_called('POST', '/servers/1234/remote-consoles')
vc = self.cs.servers.get_vnc_console(s, 'fake') vc = self.cs.servers.get_vnc_console(s, 'novnc')
self.assert_request_id(vc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(vc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles') self.assert_called('POST', '/servers/1234/remote-consoles')
# test the case with invalid console type
self.assertRaises(exceptions.UnsupportedConsoleType,
s.get_vnc_console,
'invalid')
def test_get_spice_console(self): def test_get_spice_console(self):
s = self.cs.servers.get(1234) s = self.cs.servers.get(1234)
sc = s.get_spice_console('fake') sc = s.get_spice_console('spice-html5')
self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles') self.assert_called('POST', '/servers/1234/remote-consoles')
sc = self.cs.servers.get_spice_console(s, 'fake') sc = self.cs.servers.get_spice_console(s, 'spice-html5')
self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles') self.assert_called('POST', '/servers/1234/remote-consoles')
# test the case with invalid console type
self.assertRaises(exceptions.UnsupportedConsoleType,
s.get_spice_console,
'invalid')
def test_get_serial_console(self): def test_get_serial_console(self):
s = self.cs.servers.get(1234) s = self.cs.servers.get(1234)
sc = s.get_serial_console('fake') sc = s.get_serial_console('serial')
self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles') self.assert_called('POST', '/servers/1234/remote-consoles')
sc = self.cs.servers.get_serial_console(s, 'fake') sc = self.cs.servers.get_serial_console(s, 'serial')
self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(sc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles') self.assert_called('POST', '/servers/1234/remote-consoles')
# test the case with invalid console type
self.assertRaises(exceptions.UnsupportedConsoleType,
s.get_serial_console,
'invalid')
def test_get_rdp_console(self): def test_get_rdp_console(self):
s = self.cs.servers.get(1234) s = self.cs.servers.get(1234)
rc = s.get_rdp_console('fake') rc = s.get_rdp_console('rdp-html5')
self.assert_request_id(rc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(rc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles') self.assert_called('POST', '/servers/1234/remote-consoles')
rc = self.cs.servers.get_rdp_console(s, 'fake') rc = self.cs.servers.get_rdp_console(s, 'rdp-html5')
self.assert_request_id(rc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(rc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles') self.assert_called('POST', '/servers/1234/remote-consoles')
# test the case with invalid console type
self.assertRaises(exceptions.UnsupportedConsoleType,
s.get_rdp_console,
'invalid')
def test_get_console_url(self):
s = self.cs.servers.get(1234)
vc = s.get_console_url('novnc')
self.assert_request_id(vc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles')
vc = self.cs.servers.get_console_url(s, 'novnc')
self.assert_request_id(vc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles')
# test the case with invalid console type
self.assertRaises(exceptions.UnsupportedConsoleType,
s.get_console_url,
'invalid')
# console type webmks is supported since api version 2.8
self.assertRaises(exceptions.UnsupportedConsoleType,
s.get_console_url,
'webmks')
class ServersV28Test(ServersV26Test): class ServersV28Test(ServersV26Test):
@ -1054,6 +1108,21 @@ class ServersV28Test(ServersV26Test):
self.assert_request_id(mksc, fakes.FAKE_REQUEST_ID_LIST) self.assert_request_id(mksc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles') self.assert_called('POST', '/servers/1234/remote-consoles')
def test_get_console_url(self):
s = self.cs.servers.get(1234)
mksc = s.get_console_url('novnc')
self.assert_request_id(mksc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles')
mksc = self.cs.servers.get_console_url(s, 'novnc')
self.assert_request_id(mksc, fakes.FAKE_REQUEST_ID_LIST)
self.assert_called('POST', '/servers/1234/remote-consoles')
# test the case with invalid console type
self.assertRaises(exceptions.UnsupportedConsoleType,
s.get_console_url,
'invalid')
class ServersV214Test(ServersV28Test): class ServersV214Test(ServersV28Test):

View File

@ -35,6 +35,23 @@ from novaclient.v2 import security_groups
REBOOT_SOFT, REBOOT_HARD = 'SOFT', 'HARD' REBOOT_SOFT, REBOOT_HARD = 'SOFT', 'HARD'
CONSOLE_TYPE_ACTION_MAPPING = {
'novnc': 'os-getVNCConsole',
'xvpvnc': 'os-getVNCConsole',
'spice-html5': 'os-getSPICEConsole',
'rdp-html5': 'os-getRDPConsole',
'serial': 'os-getSerialConsole'
}
CONSOLE_TYPE_PROTOCOL_MAPPING = {
'novnc': 'vnc',
'xvpvnc': 'vnc',
'spice-html5': 'spice',
'rdp-html5': 'rdp',
'serial': 'serial',
'webmks': 'mks'
}
class Server(base.Resource): class Server(base.Resource):
HUMAN_ID = True HUMAN_ID = True
@ -121,6 +138,15 @@ class Server(base.Resource):
""" """
return self.manager.get_mks_console(self) return self.manager.get_mks_console(self)
def get_console_url(self, console_type):
"""
Retrieve a console of a particular protocol and console_type
:param console_type: Type of console
"""
return self.manager.get_console_url(self, console_type)
def get_password(self, private_key=None): def get_password(self, private_key=None):
""" """
Get password for a Server. Get password for a Server.
@ -904,7 +930,7 @@ class ServerManager(base.BootingManagerWithFind):
:returns: An instance of novaclient.base.DictWithMeta :returns: An instance of novaclient.base.DictWithMeta
""" """
return self._action('os-getVNCConsole', server, {'type': console_type}) return self.get_console_url(server, console_type)
@api_versions.wraps('2.0', '2.5') @api_versions.wraps('2.0', '2.5')
def get_spice_console(self, server, console_type): def get_spice_console(self, server, console_type):
@ -916,8 +942,7 @@ class ServerManager(base.BootingManagerWithFind):
:returns: An instance of novaclient.base.DictWithMeta :returns: An instance of novaclient.base.DictWithMeta
""" """
return self._action('os-getSPICEConsole', server, return self.get_console_url(server, console_type)
{'type': console_type})
@api_versions.wraps('2.0', '2.5') @api_versions.wraps('2.0', '2.5')
def get_rdp_console(self, server, console_type): def get_rdp_console(self, server, console_type):
@ -929,8 +954,7 @@ class ServerManager(base.BootingManagerWithFind):
:returns: An instance of novaclient.base.DictWithMeta :returns: An instance of novaclient.base.DictWithMeta
""" """
return self._action('os-getRDPConsole', server, return self.get_console_url(server, console_type)
{'type': console_type})
@api_versions.wraps('2.0', '2.5') @api_versions.wraps('2.0', '2.5')
def get_serial_console(self, server, console_type): def get_serial_console(self, server, console_type):
@ -942,8 +966,29 @@ class ServerManager(base.BootingManagerWithFind):
:returns: An instance of novaclient.base.DictWithMeta :returns: An instance of novaclient.base.DictWithMeta
""" """
return self._action('os-getSerialConsole', server, return self.get_console_url(server, console_type)
{'type': console_type})
def _get_protocol(self, console_type):
protocol = CONSOLE_TYPE_PROTOCOL_MAPPING.get(console_type)
if not protocol:
raise exceptions.UnsupportedConsoleType(console_type)
return protocol
@api_versions.wraps('2.0', '2.5')
def get_console_url(self, server, console_type):
"""
Retrieve a console url of a server.
:param server: server to get console url for
:param console_type: type can be novnc, xvpvnc, spice-html5,
rdp-html5 and serial.
"""
action = CONSOLE_TYPE_ACTION_MAPPING.get(console_type)
if not action:
raise exceptions.UnsupportedConsoleType(console_type)
return self._action(action, server, {'type': console_type})
@api_versions.wraps('2.6') @api_versions.wraps('2.6')
def get_vnc_console(self, server, console_type): def get_vnc_console(self, server, console_type):
@ -955,8 +1000,7 @@ class ServerManager(base.BootingManagerWithFind):
:returns: An instance of novaclient.base.DictWithMeta :returns: An instance of novaclient.base.DictWithMeta
""" """
return self._console(server, return self.get_console_url(server, console_type)
{'protocol': 'vnc', 'type': console_type})
@api_versions.wraps('2.6') @api_versions.wraps('2.6')
def get_spice_console(self, server, console_type): def get_spice_console(self, server, console_type):
@ -968,8 +1012,7 @@ class ServerManager(base.BootingManagerWithFind):
:returns: An instance of novaclient.base.DictWithMeta :returns: An instance of novaclient.base.DictWithMeta
""" """
return self._console(server, return self.get_console_url(server, console_type)
{'protocol': 'spice', 'type': console_type})
@api_versions.wraps('2.6') @api_versions.wraps('2.6')
def get_rdp_console(self, server, console_type): def get_rdp_console(self, server, console_type):
@ -981,8 +1024,7 @@ class ServerManager(base.BootingManagerWithFind):
:returns: An instance of novaclient.base.DictWithMeta :returns: An instance of novaclient.base.DictWithMeta
""" """
return self._console(server, return self.get_console_url(server, console_type)
{'protocol': 'rdp', 'type': console_type})
@api_versions.wraps('2.6') @api_versions.wraps('2.6')
def get_serial_console(self, server, console_type): def get_serial_console(self, server, console_type):
@ -994,8 +1036,7 @@ class ServerManager(base.BootingManagerWithFind):
:returns: An instance of novaclient.base.DictWithMeta :returns: An instance of novaclient.base.DictWithMeta
""" """
return self._console(server, return self.get_console_url(server, console_type)
{'protocol': 'serial', 'type': console_type})
@api_versions.wraps('2.8') @api_versions.wraps('2.8')
def get_mks_console(self, server): def get_mks_console(self, server):
@ -1006,8 +1047,30 @@ class ServerManager(base.BootingManagerWithFind):
:returns: An instance of novaclient.base.DictWithMeta :returns: An instance of novaclient.base.DictWithMeta
""" """
return self._console(server, return self.get_console_url(server, 'webmks')
{'protocol': 'mks', 'type': 'webmks'})
@api_versions.wraps('2.6')
def get_console_url(self, server, console_type):
"""
Retrieve a console url of a server.
:param server: server to get console url for
:param console_type: type can be novnc/xvpvnc for protocol vnc;
spice-html5 for protocol spice; rdp-html5 for
protocol rdp; serial for protocol serial.
webmks for protocol mks (since version 2.8).
"""
if self.api_version < api_versions.APIVersion('2.8'):
if console_type == 'webmks':
raise exceptions.UnsupportedConsoleType(console_type)
protocol = self._get_protocol(console_type)
body = {'remote_console': {'protocol': protocol,
'type': console_type}}
url = '/servers/%s/remote-consoles' % base.getid(server)
resp, body = self.api.client.post(url, body=body)
return self.convert_into_with_meta(body, resp)
def get_password(self, server, private_key=None): def get_password(self, server, private_key=None):
""" """
@ -1855,15 +1918,6 @@ class ServerManager(base.BootingManagerWithFind):
url = '/servers/%s/action' % base.getid(server) url = '/servers/%s/action' % base.getid(server)
return self.api.client.post(url, body=body) return self.api.client.post(url, body=body)
def _console(self, server, info=None, **kwargs):
"""
Retrieve a console of a particular protocol -- vnc/spice/rdp/serial
"""
body = {'remote_console': info}
url = '/servers/%s/remote-consoles' % base.getid(server)
resp, body = self.api.client.post(url, body=body)
return self.convert_into_with_meta(body, resp)
@api_versions.wraps('2.26') @api_versions.wraps('2.26')
def tag_list(self, server): def tag_list(self, server):
""" """

View File

@ -0,0 +1,7 @@
---
features:
- Provides a public unified interface 'get_console_url' for classes
'novaclient.v2.servers.Server' and 'novaclient.v2.servers.ServerManager'.
Users (Heat, OpenStack-client and etc.) can call this public interface
instead of calling the individual methods to retrieve a console url
of a particular protocol.