diff --git a/sushy_tools/emulator/api_utils.py b/sushy_tools/emulator/api_utils.py new file mode 100644 index 00000000..6a840343 --- /dev/null +++ b/sushy_tools/emulator/api_utils.py @@ -0,0 +1,75 @@ +# 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. + +import functools + +import flask + + +def debug(*args, **kwargs): + flask.current_app.logger.debug(*args, **kwargs) + + +def info(*args, **kwargs): + flask.current_app.logger.info(*args, **kwargs) + + +def warning(*args, **kwargs): + flask.current_app.logger.warning(*args, **kwargs) + + +def error(*args, **kwargs): + flask.current_app.logger.error(*args, **kwargs) + + +def instance_denied(**kwargs): + deny = True + + try: + deny = (kwargs['identity'] not in + flask.current_app.config['SUSHY_EMULATOR_ALLOWED_INSTANCES']) + + except KeyError: + deny = False + + finally: + if deny: + warning('Instance %s access denied', kwargs.get('identity')) + + return deny + + +def ensure_instance_access(decorated_func): + @functools.wraps(decorated_func) + def decorator(*args, **kwargs): + if instance_denied(**kwargs): + raise error.NotFound() + + return decorated_func(*args, **kwargs) + + return decorator + + +def returns_json(decorated_func): + @functools.wraps(decorated_func) + def decorator(*args, **kwargs): + response = decorated_func(*args, **kwargs) + if isinstance(response, flask.Response): + return response + if isinstance(response, tuple): + contents, status = response + else: + contents, status = response, 200 + return flask.Response(response=contents, status=status, + content_type='application/json') + + return decorator diff --git a/sushy_tools/emulator/controllers/__init__.py b/sushy_tools/emulator/controllers/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/sushy_tools/emulator/controllers/virtual_media.py b/sushy_tools/emulator/controllers/virtual_media.py new file mode 100644 index 00000000..3b027eb8 --- /dev/null +++ b/sushy_tools/emulator/controllers/virtual_media.py @@ -0,0 +1,182 @@ +# 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. + +import flask + +from sushy_tools.emulator import api_utils +from sushy_tools import error + + +virtual_media = flask.Blueprint( + 'VirtualMedia', __name__, + url_prefix='/redfish/v1/Managers//VirtualMedia') + + +@virtual_media.route('/', methods=['GET']) +@api_utils.returns_json +def virtual_media_collection_resource(identity): + api_utils.debug('Serving virtual media resources for manager "%s"', + identity) + + return flask.render_template( + 'virtual_media_collection.json', + identity=identity, + uuid=flask.current_app.managers.get_manager(identity)['UUID'], + devices=flask.current_app.vmedia.devices + ) + + +@virtual_media.route('/', methods=['GET']) +@api_utils.returns_json +def virtual_media_resource(identity, device): + device_name = flask.current_app.vmedia.get_device_name( + identity, device) + + media_types = flask.current_app.vmedia.get_device_media_types( + identity, device) + + device_info = flask.current_app.vmedia.get_device_image_info( + identity, device) + + api_utils.debug('Serving virtual media %s at manager "%s"', + device, identity) + + return flask.render_template( + 'virtual_media.json', + identity=identity, + device=device, + name=device_name, + media_types=media_types, + image_url=device_info.image_url, + image_name=device_info.image_name, + inserted=device_info.inserted, + write_protected=device_info.write_protected, + username=device_info.username, + password=device_info.password, + verify_certificate=device_info.verify, + ) + + +@virtual_media.route('/', methods=['PATCH']) +@api_utils.returns_json +def virtual_media_patch(identity, device): + if not flask.request.json: + raise error.BadRequest("Empty or malformed patch") + + api_utils.debug('Updating virtual media %s at manager "%s"', + device, identity) + + verify = flask.request.json.get('VerifyCertificate') + if verify is not None: + if not isinstance(verify, bool): + raise error.BadRequest("VerifyCertificate must be a boolean") + + flask.current_app.vmedia.update_device_info( + identity, device, verify=verify) + return '', 204 + else: + raise error.BadRequest("Empty or malformed patch") + + +@virtual_media.route('//Certificates', methods=['GET']) +@api_utils.returns_json +def virtual_media_certificates(identity, device): + location = \ + f'/redfish/v1/Managers/{identity}/VirtualMedia/{device}/Certificates' + return flask.render_template( + 'certificate_collection.json', + location=location, + # TODO(dtantsur): implement + certificates=[], + ) + + +@virtual_media.route('//Certificates', methods=['POST']) +@api_utils.returns_json +def virtual_media_add_certificate(identity, device): + if not flask.request.json: + raise error.BadRequest("Empty or malformed certificate") + + # TODO(dtantsur): implement + raise error.NotSupportedError("Not implemented") + + +@virtual_media.route('//Actions/VirtualMedia.InsertMedia', + methods=['POST']) +@api_utils.returns_json +def virtual_media_insert(identity, device): + image = flask.request.json.get('Image') + inserted = flask.request.json.get('Inserted', True) + write_protected = flask.request.json.get('WriteProtected', True) + username = flask.request.json.get('UserName', '') + password = flask.request.json.get('Password', '') + + if (not username and password) or (username and not password): + message = "UserName and Password must be passed together" + return flask.render_template('error.json', message=message), 400 + + manager = flask.current_app.managers.get_manager(identity) + systems = flask.current_app.managers.get_managed_systems(manager) + if not systems: + api_utils.warning('Manager %s manages no systems', identity) + return '', 204 + + image_path = flask.current_app.vmedia.insert_image( + identity, device, image, inserted, write_protected, + username=username, password=password) + + for system in systems: + try: + flask.current_app.systems.set_boot_image( + system, device, boot_image=image_path, + write_protected=write_protected) + + except error.NotSupportedError as ex: + api_utils.warning( + 'System %s failed to set boot image %s on device %s: ' + '%s', system, image_path, device, ex) + + api_utils.info( + 'Virtual media placed into device %(dev)s of manager %(mgr)s for ' + 'systems %(sys)s. Image %(img)s inserted %(ins)s', + {'dev': device, 'mgr': identity, 'sys': systems, + 'img': image or '', 'ins': inserted}) + + return '', 204 + + +@virtual_media.route('//Actions/VirtualMedia.EjectMedia', + methods=['POST']) +@api_utils.returns_json +def virtual_media_eject(identity, device): + flask.current_app.vmedia.eject_image(identity, device) + + manager = flask.current_app.managers.get_manager(identity) + systems = flask.current_app.managers.get_managed_systems(manager) + if not systems: + api_utils.warning('Manager %s manages no systems', identity) + return '', 204 + + for system in systems: + try: + flask.current_app.systems.set_boot_image(system, device) + + except error.NotSupportedError as ex: + api_utils.warning( + 'System %s failed to remove boot image from device %s: ' + '%s', system, device, ex) + + api_utils.info( + 'Virtual media ejected from device %s manager %s systems %s', + device, identity, systems) + + return '', 204 diff --git a/sushy_tools/emulator/main.py b/sushy_tools/emulator/main.py index 98ecc915..f199b373 100755 --- a/sushy_tools/emulator/main.py +++ b/sushy_tools/emulator/main.py @@ -15,7 +15,6 @@ import argparse from datetime import datetime -import functools import json import os import ssl @@ -25,6 +24,8 @@ import flask from ironic_lib import auth_basic from werkzeug import exceptions as wz_exc +from sushy_tools.emulator import api_utils +from sushy_tools.emulator.controllers import virtual_media as vmctl from sushy_tools.emulator import memoize from sushy_tools.emulator.resources import chassis as chsdriver from sushy_tools.emulator.resources import drives as drvdriver @@ -152,55 +153,11 @@ class Application(flask.Flask): app = Application() - - -def instance_denied(**kwargs): - deny = True - - try: - deny = (kwargs['identity'] not in - app.config['SUSHY_EMULATOR_ALLOWED_INSTANCES']) - - except KeyError: - deny = False - - finally: - if deny: - app.logger.warning('Instance %s access denied', - kwargs.get('identity')) - - return deny - - -def ensure_instance_access(decorated_func): - @functools.wraps(decorated_func) - def decorator(*args, **kwargs): - if instance_denied(**kwargs): - raise error.NotFound() - - return decorated_func(*args, **kwargs) - - return decorator - - -def returns_json(decorated_func): - @functools.wraps(decorated_func) - def decorator(*args, **kwargs): - response = decorated_func(*args, **kwargs) - if isinstance(response, flask.Response): - return response - if isinstance(response, tuple): - contents, status = response - else: - contents, status = response, 200 - return flask.Response(response=contents, status=status, - content_type='application/json') - - return decorator +app.register_blueprint(vmctl.virtual_media) @app.errorhandler(Exception) -@returns_json +@api_utils.returns_json def all_exception_handler(message): if isinstance(message, error.AliasAccessError): url = flask.url_for(flask.request.endpoint, identity=message.args[0]) @@ -219,13 +176,13 @@ def all_exception_handler(message): @app.route('/redfish/v1/') -@returns_json +@api_utils.returns_json def root_resource(): return flask.render_template('root.json') @app.route('/redfish/v1/Chassis') -@returns_json +@api_utils.returns_json def chassis_collection_resource(): app.logger.debug('Serving chassis list') @@ -236,7 +193,7 @@ def chassis_collection_resource(): @app.route('/redfish/v1/Chassis/', methods=['GET', 'PATCH']) -@returns_json +@api_utils.returns_json def chassis_resource(identity): chassis = app.chassis @@ -288,7 +245,7 @@ def chassis_resource(identity): @app.route('/redfish/v1/Chassis//Thermal', methods=['GET']) -@returns_json +@api_utils.returns_json def thermal_resource(identity): chassis = app.chassis @@ -312,7 +269,7 @@ def thermal_resource(identity): @app.route('/redfish/v1/Managers') -@returns_json +@api_utils.returns_json def manager_collection_resource(): app.logger.debug('Serving managers list') @@ -335,7 +292,7 @@ def jsonify(obj_type, obj_version, obj): @app.route('/redfish/v1/Managers/', methods=['GET']) -@returns_json +@api_utils.returns_json def manager_resource(identity): app.logger.debug('Serving resources for manager "%s"', identity) @@ -381,179 +338,11 @@ def manager_resource(identity): }) -@app.route('/redfish/v1/Managers//VirtualMedia', methods=['GET']) -@returns_json -def virtual_media_collection_resource(identity): - app.logger.debug('Serving virtual media resources for ' - 'manager "%s"', identity) - - return flask.render_template( - 'virtual_media_collection.json', - identity=identity, - uuid=app.managers.get_manager(identity)['UUID'], - devices=app.vmedia.devices - ) - - -@app.route('/redfish/v1/Managers//VirtualMedia/', - methods=['GET']) -@returns_json -def virtual_media_resource(identity, device): - device_name = app.vmedia.get_device_name( - identity, device) - - media_types = app.vmedia.get_device_media_types( - identity, device) - - device_info = app.vmedia.get_device_image_info(identity, device) - - app.logger.debug('Serving virtual media %s at ' - 'manager "%s"', device, identity) - - return flask.render_template( - 'virtual_media.json', - identity=identity, - device=device, - name=device_name, - media_types=media_types, - image_url=device_info.image_url, - image_name=device_info.image_name, - inserted=device_info.inserted, - write_protected=device_info.write_protected, - username=device_info.username, - password=device_info.password, - verify_certificate=device_info.verify, - ) - - -@app.route( - '/redfish/v1/Managers//VirtualMedia/', - methods=['PATCH']) -@returns_json -def virtual_media_patch(identity, device): - if not flask.request.json: - raise error.BadRequest("Empty or malformed patch") - - app.logger.debug('Updating virtual media %s at manager "%s"', - device, identity) - - verify = flask.request.json.get('VerifyCertificate') - if verify is not None: - if not isinstance(verify, bool): - raise error.BadRequest("VerifyCertificate must be a boolean") - - app.vmedia.update_device_info(identity, device, verify=verify) - return '', 204 - else: - raise error.BadRequest("Empty or malformed patch") - - -@app.route( - '/redfish/v1/Managers//VirtualMedia//Certificates', - methods=['GET']) -@returns_json -def virtual_media_certificates(identity, device): - location = \ - f'/redfish/v1/Managers/{identity}/VirtualMedia/{device}/Certificates' - return flask.render_template( - 'certificate_collection.json', - location=location, - # TODO(dtantsur): implement - certificates=[], - ) - - -@app.route( - '/redfish/v1/Managers//VirtualMedia//Certificates', - methods=['POST']) -@returns_json -def virtual_media_add_certificate(identity, device): - if not flask.request.json: - raise error.BadRequest("Empty or malformed certificate") - - # TODO(dtantsur): implement - raise error.NotSupportedError("Not implemented") - - -@app.route('/redfish/v1/Managers//VirtualMedia/' - '/Actions/VirtualMedia.InsertMedia', - methods=['POST']) -@returns_json -def virtual_media_insert(identity, device): - image = flask.request.json.get('Image') - inserted = flask.request.json.get('Inserted', True) - write_protected = flask.request.json.get('WriteProtected', True) - username = flask.request.json.get('UserName', '') - password = flask.request.json.get('Password', '') - - if (not username and password) or (username and not password): - message = "UserName and Password must be passed together" - return flask.render_template('error.json', message=message), 400 - - manager = app.managers.get_manager(identity) - systems = app.managers.get_managed_systems(manager) - if not systems: - app.logger.warning('Manager %s manages no systems', identity) - return '', 204 - - image_path = app.vmedia.insert_image( - identity, device, image, inserted, write_protected, - username=username, password=password) - - for system in systems: - try: - app.systems.set_boot_image( - system, device, boot_image=image_path, - write_protected=write_protected) - - except error.NotSupportedError as ex: - app.logger.warning( - 'System %s failed to set boot image %s on device %s: ' - '%s', system, image_path, device, ex) - - app.logger.info( - 'Virtual media placed into device %(dev)s of manager %(mgr)s for ' - 'systems %(sys)s. Image %(img)s inserted %(ins)s', - {'dev': device, 'mgr': identity, 'sys': systems, - 'img': image or '', 'ins': inserted}) - - return '', 204 - - -@app.route('/redfish/v1/Managers//VirtualMedia/' - '/Actions/VirtualMedia.EjectMedia', - methods=['POST']) -@returns_json -def virtual_media_eject(identity, device): - app.vmedia.eject_image(identity, device) - - manager = app.managers.get_manager(identity) - systems = app.managers.get_managed_systems(manager) - if not systems: - app.logger.warning('Manager %s manages no systems', identity) - return '', 204 - - for system in systems: - try: - app.systems.set_boot_image(system, device) - - except error.NotSupportedError as ex: - app.logger.warning( - 'System %s failed to remove boot image from device %s: ' - '%s', system, device, ex) - - app.logger.info( - 'Virtual media ejected from device %s manager %s systems %s', - device, identity, systems) - - return '', 204 - - @app.route('/redfish/v1/Systems') -@returns_json +@api_utils.returns_json def system_collection_resource(): systems = [system for system in app.systems.systems - if not instance_denied(identity=system)] + if not api_utils.instance_denied(identity=system)] app.logger.debug('Serving systems list') @@ -562,8 +351,8 @@ def system_collection_resource(): @app.route('/redfish/v1/Systems/', methods=['GET', 'PATCH']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def system_resource(identity): if flask.request.method == 'GET': @@ -629,8 +418,8 @@ def system_resource(identity): @app.route('/redfish/v1/Systems//EthernetInterfaces', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def ethernet_interfaces_collection(identity): nics = app.systems.get_nics(identity) @@ -641,8 +430,8 @@ def ethernet_interfaces_collection(identity): @app.route('/redfish/v1/Systems//EthernetInterfaces/', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def ethernet_interface(identity, nic_id): nics = app.systems.get_nics(identity) @@ -656,8 +445,8 @@ def ethernet_interface(identity, nic_id): @app.route('/redfish/v1/Systems//Processors', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def processors_collection(identity): processors = app.systems.get_processors(identity) @@ -668,8 +457,8 @@ def processors_collection(identity): @app.route('/redfish/v1/Systems//Processors/', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def processor(identity, processor_id): processors = app.systems.get_processors(identity) @@ -683,8 +472,8 @@ def processor(identity, processor_id): @app.route('/redfish/v1/Systems//Actions/ComputerSystem.Reset', methods=['POST']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def system_reset_action(identity): reset_type = flask.request.json.get('ResetType') @@ -697,8 +486,8 @@ def system_reset_action(identity): @app.route('/redfish/v1/Systems//BIOS', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def bios(identity): bios = app.systems.get_bios(identity) @@ -712,8 +501,8 @@ def bios(identity): @app.route('/redfish/v1/Systems//BIOS/Settings', methods=['GET', 'PATCH']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def bios_settings(identity): if flask.request.method == 'GET': @@ -738,8 +527,8 @@ def bios_settings(identity): @app.route('/redfish/v1/Systems//BIOS/Actions/Bios.ResetBios', methods=['POST']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def system_reset_bios(identity): app.systems.reset_bios(identity) @@ -750,8 +539,8 @@ def system_reset_bios(identity): @app.route('/redfish/v1/Systems//SimpleStorage', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def simple_storage_collection(identity): simple_storage_controllers = ( app.systems.get_simple_storage_collection(identity)) @@ -763,8 +552,8 @@ def simple_storage_collection(identity): @app.route('/redfish/v1/Systems//SimpleStorage/', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def simple_storage(identity, simple_storage_id): simple_storage_controllers = ( app.systems.get_simple_storage_collection(identity)) @@ -779,8 +568,8 @@ def simple_storage(identity, simple_storage_id): @app.route('/redfish/v1/Systems//Storage', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def storage_collection(identity): uuid = app.systems.uuid(identity) @@ -793,8 +582,8 @@ def storage_collection(identity): @app.route('/redfish/v1/Systems//Storage/', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def storage(identity, storage_id): uuid = app.systems.uuid(identity) storage_col = app.storage.get_storage_col(uuid) @@ -809,8 +598,8 @@ def storage(identity, storage_id): @app.route('/redfish/v1/Systems//Storage//Drives/', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def drive_resource(identity, stg_id, drv_id): uuid = app.systems.uuid(identity) drives = app.drives.get_drives(uuid, stg_id) @@ -825,8 +614,8 @@ def drive_resource(identity, stg_id, drv_id): @app.route('/redfish/v1/Systems//Storage//Volumes', methods=['GET', 'POST']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def volumes_collection(identity, storage_id): uuid = app.systems.uuid(identity) @@ -867,8 +656,8 @@ def volumes_collection(identity, storage_id): @app.route('/redfish/v1/Systems//Storage//Volumes/', methods=['GET']) -@ensure_instance_access -@returns_json +@api_utils.ensure_instance_access +@api_utils.returns_json def volume(identity, stg_id, vol_id): uuid = app.systems.uuid(identity) vol_col = app.volumes.get_volumes_col(uuid, stg_id) @@ -887,7 +676,7 @@ def volume(identity, stg_id, vol_id): @app.route('/redfish/v1/Registries') -@returns_json +@api_utils.returns_json def registry_file_collection(): app.logger.debug('Serving registry file collection') @@ -896,7 +685,7 @@ def registry_file_collection(): @app.route('/redfish/v1/Registries/BiosAttributeRegistry.v1_0_0') -@returns_json +@api_utils.returns_json def bios_attribute_registry_file(): app.logger.debug('Serving BIOS attribute registry file') @@ -905,7 +694,7 @@ def bios_attribute_registry_file(): @app.route('/redfish/v1/Registries/Messages') -@returns_json +@api_utils.returns_json def message_registry_file(): app.logger.debug('Serving message registry file') @@ -914,7 +703,7 @@ def message_registry_file(): @app.route('/redfish/v1/Systems/Bios/BiosRegistry') -@returns_json +@api_utils.returns_json def bios_registry(): app.logger.debug('Serving BIOS registry') @@ -922,7 +711,7 @@ def bios_registry(): @app.route('/redfish/v1/Registries/Messages/Registry') -@returns_json +@api_utils.returns_json def message_registry(): app.logger.debug('Serving message registry') diff --git a/sushy_tools/tests/unit/emulator/controllers/__init__.py b/sushy_tools/tests/unit/emulator/controllers/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/sushy_tools/tests/unit/emulator/controllers/test_virtual_media.py b/sushy_tools/tests/unit/emulator/controllers/test_virtual_media.py new file mode 100644 index 00000000..7d32e347 --- /dev/null +++ b/sushy_tools/tests/unit/emulator/controllers/test_virtual_media.py @@ -0,0 +1,168 @@ +# 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 sushy_tools.emulator.resources import vmedia +from sushy_tools import error +from sushy_tools.tests.unit.emulator import test_main + + +@test_main.patch_resource('vmedia') +@test_main.patch_resource('managers') +class VirtualMediaTestCase(test_main.EmulatorTestCase): + + def test_virtual_media_collection(self, managers_mock, vmedia_mock): + managers_mock = managers_mock.return_value + managers_mock.managers = [self.uuid] + managers_mock.get_manager.return_value = {'UUID': self.uuid} + vmedia_mock.return_value.devices = ['CD', 'Floppy'] + + response = self.app.get( + 'redfish/v1/Managers/%s/VirtualMedia' % self.uuid) + + self.assertEqual(200, response.status_code) + self.assertEqual('Virtual Media Services', response.json['Name']) + self.assertEqual(2, response.json['Members@odata.count']) + self.assertEqual( + ['/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid, + '/redfish/v1/Managers/%s/VirtualMedia/Floppy' % self.uuid], + [m['@odata.id'] for m in response.json['Members']]) + + def test_virtual_media_collection_empty(self, managers_mock, vmedia_mock): + vmedia_mock.return_value.get_devices.return_value = [] + + response = self.app.get( + 'redfish/v1/Managers/' + self.uuid + '/VirtualMedia') + + self.assertEqual(200, response.status_code) + self.assertEqual('Virtual Media Services', response.json['Name']) + self.assertEqual(0, response.json['Members@odata.count']) + self.assertEqual([], response.json['Members']) + + def test_virtual_media(self, managers_mock, vmedia_mock): + vmedia_mock = vmedia_mock.return_value + vmedia_mock.get_device_name.return_value = 'CD' + vmedia_mock.get_device_media_types.return_value = [ + 'CD', 'DVD'] + vmedia_mock.get_device_image_info.return_value = vmedia.DeviceInfo( + 'image-of-a-fish', 'fishy.iso', True, True, '', '', False) + + response = self.app.get( + '/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid) + + self.assertEqual(200, response.status_code, response.json) + self.assertEqual('CD', response.json['Id']) + self.assertEqual(['CD', 'DVD'], response.json['MediaTypes']) + self.assertEqual('fishy.iso', response.json['Image']) + self.assertEqual('image-of-a-fish', response.json['ImageName']) + self.assertTrue(response.json['Inserted']) + self.assertTrue(response.json['WriteProtected']) + self.assertEqual('', response.json['UserName']) + self.assertEqual('', response.json['Password']) + self.assertFalse(response.json['VerifyCertificate']) + self.assertEqual( + '/redfish/v1/Managers/%s/VirtualMedia/CD/Certificates' % self.uuid, + response.json['Certificates']['@odata.id']) + + def test_virtual_media_with_auth(self, managers_mock, vmedia_mock): + vmedia_mock = vmedia_mock.return_value + vmedia_mock.get_device_name.return_value = 'CD' + vmedia_mock.get_device_media_types.return_value = [ + 'CD', 'DVD'] + vmedia_mock.get_device_image_info.return_value = vmedia.DeviceInfo( + 'image-of-a-fish', 'fishy.iso', True, True, 'Admin', 'Secret', + False) + + response = self.app.get( + '/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid) + + self.assertEqual(200, response.status_code, response.json) + self.assertEqual('CD', response.json['Id']) + self.assertEqual(['CD', 'DVD'], response.json['MediaTypes']) + self.assertEqual('fishy.iso', response.json['Image']) + self.assertEqual('image-of-a-fish', response.json['ImageName']) + self.assertTrue(response.json['Inserted']) + self.assertTrue(response.json['WriteProtected']) + self.assertEqual('Admin', response.json['UserName']) + self.assertEqual('******', response.json['Password']) + self.assertFalse(response.json['VerifyCertificate']) + + def test_virtual_media_not_found(self, managers_mock, vmedia_mock): + vmedia_mock.return_value.get_device_name.side_effect = error.NotFound + + response = self.app.get( + '/redfish/v1/Managers/%s/VirtualMedia/DVD-ROM' % self.uuid) + + self.assertEqual(404, response.status_code) + + def test_virtual_media_update(self, managers_mock, vmedia_mock): + response = self.app.patch( + '/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid, + json={'VerifyCertificate': True}) + + self.assertEqual(204, response.status_code) + vmedia_mock = vmedia_mock.return_value + vmedia_mock.update_device_info.assert_called_once_with( + self.uuid, 'CD', verify=True) + + def test_virtual_media_update_not_found(self, managers_mock, vmedia_mock): + vmedia_mock = vmedia_mock.return_value + vmedia_mock.update_device_info.side_effect = error.NotFound + + response = self.app.patch( + '/redfish/v1/Managers/%s/VirtualMedia/DVD-ROM' % self.uuid, + json={'VerifyCertificate': True}) + + self.assertEqual(404, response.status_code) + + def test_virtual_media_update_invalid(self, managers_mock, vmedia_mock): + response = self.app.patch( + '/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid, + json={'VerifyCertificate': 'banana'}) + + self.assertEqual(400, response.status_code) + + def test_virtual_media_update_empty(self, managers_mock, vmedia_mock): + response = self.app.patch( + '/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid) + + self.assertEqual(400, response.status_code) + + def test_virtual_media_insert(self, managers_mock, vmedia_mock): + response = self.app.post( + '/redfish/v1/Managers/%s/VirtualMedia/CD/Actions/' + 'VirtualMedia.InsertMedia' % self.uuid, + json={"Image": "http://fish.iso"}) + + self.assertEqual(204, response.status_code) + + vmedia_mock.return_value.insert_image.called_once_with( + 'CD', 'http://fish.iso', True, False) + + def test_virtual_media_eject(self, managers_mock, vmedia_mock): + response = self.app.post( + '/redfish/v1/Managers/%s/VirtualMedia/CD/Actions/' + 'VirtualMedia.EjectMedia' % self.uuid, + json={}) + + self.assertEqual(204, response.status_code) + + vmedia_mock.return_value.eject_image.called_once_with('CD') + + def test_virtual_media_certificates(self, managers_mock, vmedia_mock): + response = self.app.get( + '/redfish/v1/Managers/%s/VirtualMedia/CD/Certificates' % self.uuid) + + self.assertEqual(200, response.status_code, response.json) + self.assertEqual(0, response.json['Members@odata.count']) + self.assertEqual([], response.json['Members']) + self.assertEqual(['PEM'], + response.json['@Redfish.SupportedCertificates']) diff --git a/sushy_tools/tests/unit/emulator/test_api_utils.py b/sushy_tools/tests/unit/emulator/test_api_utils.py new file mode 100644 index 00000000..032c0faf --- /dev/null +++ b/sushy_tools/tests/unit/emulator/test_api_utils.py @@ -0,0 +1,49 @@ +# 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 unittest import mock + +from sushy_tools.emulator import api_utils +from sushy_tools.emulator import main +from sushy_tools.tests.unit.emulator import test_main + + +class InstanceDeniedTestCase(test_main.EmulatorTestCase): + + def setUp(self): + super().setUp() + ctx = main.app.app_context().__enter__() + self.addCleanup(lambda: ctx.__exit__(None, None, None)) + + @mock.patch.dict(main.app.config, {}, clear=True) + def test_instance_denied_allow_all(self): + self.assertFalse(api_utils.instance_denied(identity='x')) + + @mock.patch.dict( + main.app.config, {'SUSHY_EMULATOR_ALLOWED_INSTANCES': {}}) + def test_instance_denied_disallow_all(self): + self.assertTrue(api_utils.instance_denied(identity='a')) + + def test_instance_denied_undefined_option(self): + with mock.patch.dict(main.app.config): + main.app.config.pop('SUSHY_EMULATOR_ALLOWED_INSTANCES', None) + self.assertFalse(api_utils.instance_denied(identity='a')) + + @mock.patch.dict( + main.app.config, {'SUSHY_EMULATOR_ALLOWED_INSTANCES': {'a'}}) + def test_instance_denied_allow_some(self): + self.assertFalse(api_utils.instance_denied(identity='a')) + + @mock.patch.dict( + main.app.config, {'SUSHY_EMULATOR_ALLOWED_INSTANCES': {'a'}}) + def test_instance_denied_disallow_some(self): + self.assertTrue(api_utils.instance_denied(identity='b')) diff --git a/sushy_tools/tests/unit/emulator/test_main.py b/sushy_tools/tests/unit/emulator/test_main.py index 1d051466..3a506650 100644 --- a/sushy_tools/tests/unit/emulator/test_main.py +++ b/sushy_tools/tests/unit/emulator/test_main.py @@ -16,7 +16,6 @@ from unittest import mock from oslotest import base from sushy_tools.emulator import main -from sushy_tools.emulator.resources import vmedia from sushy_tools import error @@ -311,33 +310,6 @@ class SystemsTestCase(EmulatorTestCase): self.assertEqual(500, response.status_code) -class InstanceDeniedTestCase(EmulatorTestCase): - - @mock.patch.dict(main.app.config, {}, clear=True) - def test_instance_denied_allow_all(self): - self.assertFalse(main.instance_denied(identity='x')) - - @mock.patch.dict( - main.app.config, {'SUSHY_EMULATOR_ALLOWED_INSTANCES': {}}) - def test_instance_denied_disallow_all(self): - self.assertTrue(main.instance_denied(identity='a')) - - def test_instance_denied_undefined_option(self): - with mock.patch.dict(main.app.config): - main.app.config.pop('SUSHY_EMULATOR_ALLOWED_INSTANCES', None) - self.assertFalse(main.instance_denied(identity='a')) - - @mock.patch.dict( - main.app.config, {'SUSHY_EMULATOR_ALLOWED_INSTANCES': {'a'}}) - def test_instance_denied_allow_some(self): - self.assertFalse(main.instance_denied(identity='a')) - - @mock.patch.dict( - main.app.config, {'SUSHY_EMULATOR_ALLOWED_INSTANCES': {'a'}}) - def test_instance_denied_disallow_some(self): - self.assertTrue(main.instance_denied(identity='b')) - - @patch_resource('systems') class BiosTestCase(EmulatorTestCase): @@ -455,159 +427,6 @@ class EthernetInterfacesTestCase(EmulatorTestCase): self.assertEqual(404, response.status_code) -@patch_resource('vmedia') -@patch_resource('managers') -class VirtualMediaTestCase(EmulatorTestCase): - - def test_virtual_media_collection(self, managers_mock, vmedia_mock): - managers_mock = managers_mock.return_value - managers_mock.managers = [self.uuid] - managers_mock.get_manager.return_value = {'UUID': self.uuid} - vmedia_mock.return_value.devices = ['CD', 'Floppy'] - - response = self.app.get( - 'redfish/v1/Managers/%s/VirtualMedia' % self.uuid) - - self.assertEqual(200, response.status_code) - self.assertEqual('Virtual Media Services', response.json['Name']) - self.assertEqual(2, response.json['Members@odata.count']) - self.assertEqual( - ['/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid, - '/redfish/v1/Managers/%s/VirtualMedia/Floppy' % self.uuid], - [m['@odata.id'] for m in response.json['Members']]) - - def test_virtual_media_collection_empty(self, managers_mock, vmedia_mock): - vmedia_mock.return_value.get_devices.return_value = [] - - response = self.app.get( - 'redfish/v1/Managers/' + self.uuid + '/VirtualMedia') - - self.assertEqual(200, response.status_code) - self.assertEqual('Virtual Media Services', response.json['Name']) - self.assertEqual(0, response.json['Members@odata.count']) - self.assertEqual([], response.json['Members']) - - def test_virtual_media(self, managers_mock, vmedia_mock): - vmedia_mock = vmedia_mock.return_value - vmedia_mock.get_device_name.return_value = 'CD' - vmedia_mock.get_device_media_types.return_value = [ - 'CD', 'DVD'] - vmedia_mock.get_device_image_info.return_value = vmedia.DeviceInfo( - 'image-of-a-fish', 'fishy.iso', True, True, '', '', False) - - response = self.app.get( - '/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid) - - self.assertEqual(200, response.status_code, response.json) - self.assertEqual('CD', response.json['Id']) - self.assertEqual(['CD', 'DVD'], response.json['MediaTypes']) - self.assertEqual('fishy.iso', response.json['Image']) - self.assertEqual('image-of-a-fish', response.json['ImageName']) - self.assertTrue(response.json['Inserted']) - self.assertTrue(response.json['WriteProtected']) - self.assertEqual('', response.json['UserName']) - self.assertEqual('', response.json['Password']) - self.assertFalse(response.json['VerifyCertificate']) - self.assertEqual( - '/redfish/v1/Managers/%s/VirtualMedia/CD/Certificates' % self.uuid, - response.json['Certificates']['@odata.id']) - - def test_virtual_media_with_auth(self, managers_mock, vmedia_mock): - vmedia_mock = vmedia_mock.return_value - vmedia_mock.get_device_name.return_value = 'CD' - vmedia_mock.get_device_media_types.return_value = [ - 'CD', 'DVD'] - vmedia_mock.get_device_image_info.return_value = vmedia.DeviceInfo( - 'image-of-a-fish', 'fishy.iso', True, True, 'Admin', 'Secret', - False) - - response = self.app.get( - '/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid) - - self.assertEqual(200, response.status_code, response.json) - self.assertEqual('CD', response.json['Id']) - self.assertEqual(['CD', 'DVD'], response.json['MediaTypes']) - self.assertEqual('fishy.iso', response.json['Image']) - self.assertEqual('image-of-a-fish', response.json['ImageName']) - self.assertTrue(response.json['Inserted']) - self.assertTrue(response.json['WriteProtected']) - self.assertEqual('Admin', response.json['UserName']) - self.assertEqual('******', response.json['Password']) - self.assertFalse(response.json['VerifyCertificate']) - - def test_virtual_media_not_found(self, managers_mock, vmedia_mock): - vmedia_mock.return_value.get_device_name.side_effect = error.NotFound - - response = self.app.get( - '/redfish/v1/Managers/%s/VirtualMedia/DVD-ROM' % self.uuid) - - self.assertEqual(404, response.status_code) - - def test_virtual_media_update(self, managers_mock, vmedia_mock): - response = self.app.patch( - '/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid, - json={'VerifyCertificate': True}) - - self.assertEqual(204, response.status_code) - vmedia_mock = vmedia_mock.return_value - vmedia_mock.update_device_info.assert_called_once_with( - self.uuid, 'CD', verify=True) - - def test_virtual_media_update_not_found(self, managers_mock, vmedia_mock): - vmedia_mock = vmedia_mock.return_value - vmedia_mock.update_device_info.side_effect = error.NotFound - - response = self.app.patch( - '/redfish/v1/Managers/%s/VirtualMedia/DVD-ROM' % self.uuid, - json={'VerifyCertificate': True}) - - self.assertEqual(404, response.status_code) - - def test_virtual_media_update_invalid(self, managers_mock, vmedia_mock): - response = self.app.patch( - '/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid, - json={'VerifyCertificate': 'banana'}) - - self.assertEqual(400, response.status_code) - - def test_virtual_media_update_empty(self, managers_mock, vmedia_mock): - response = self.app.patch( - '/redfish/v1/Managers/%s/VirtualMedia/CD' % self.uuid) - - self.assertEqual(400, response.status_code) - - def test_virtual_media_insert(self, managers_mock, vmedia_mock): - response = self.app.post( - '/redfish/v1/Managers/%s/VirtualMedia/CD/Actions/' - 'VirtualMedia.InsertMedia' % self.uuid, - json={"Image": "http://fish.iso"}) - - self.assertEqual(204, response.status_code) - - vmedia_mock.return_value.insert_image.called_once_with( - 'CD', 'http://fish.iso', True, False) - - def test_virtual_media_eject(self, managers_mock, vmedia_mock): - response = self.app.post( - '/redfish/v1/Managers/%s/VirtualMedia/CD/Actions/' - 'VirtualMedia.EjectMedia' % self.uuid, - json={}) - - self.assertEqual(204, response.status_code) - - vmedia_mock.return_value.eject_image.called_once_with('CD') - - def test_virtual_media_certificates(self, managers_mock, vmedia_mock): - response = self.app.get( - '/redfish/v1/Managers/%s/VirtualMedia/CD/Certificates' % self.uuid) - - self.assertEqual(200, response.status_code, response.json) - self.assertEqual(0, response.json['Members@odata.count']) - self.assertEqual([], response.json['Members']) - self.assertEqual(['PEM'], - response.json['@Redfish.SupportedCertificates']) - - @patch_resource('systems') class StorageTestCase(EmulatorTestCase):