Move virtual media endpoint handlers to a separate blueprint

The size of main.py is going out of control, let's start using Flask
blueprints to split it into clear blocks.

Change-Id: I85e09eb9647e412a9dd36a166dcd5a38328b0c64
This commit is contained in:
Dmitry Tantsur 2021-08-30 16:30:45 +02:00
parent c6a42b999a
commit c37ffb959d
8 changed files with 523 additions and 441 deletions

View File

@ -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

View File

@ -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/<identity>/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('/<device>', 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('/<device>', 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('/<device>/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('/<device>/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('/<device>/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 '<empty>', 'ins': inserted})
return '', 204
@virtual_media.route('/<device>/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

View File

@ -15,7 +15,6 @@
import argparse import argparse
from datetime import datetime from datetime import datetime
import functools
import json import json
import os import os
import ssl import ssl
@ -25,6 +24,8 @@ import flask
from ironic_lib import auth_basic from ironic_lib import auth_basic
from werkzeug import exceptions as wz_exc 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 import memoize
from sushy_tools.emulator.resources import chassis as chsdriver from sushy_tools.emulator.resources import chassis as chsdriver
from sushy_tools.emulator.resources import drives as drvdriver from sushy_tools.emulator.resources import drives as drvdriver
@ -152,55 +153,11 @@ class Application(flask.Flask):
app = Application() app = Application()
app.register_blueprint(vmctl.virtual_media)
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.errorhandler(Exception) @app.errorhandler(Exception)
@returns_json @api_utils.returns_json
def all_exception_handler(message): def all_exception_handler(message):
if isinstance(message, error.AliasAccessError): if isinstance(message, error.AliasAccessError):
url = flask.url_for(flask.request.endpoint, identity=message.args[0]) url = flask.url_for(flask.request.endpoint, identity=message.args[0])
@ -219,13 +176,13 @@ def all_exception_handler(message):
@app.route('/redfish/v1/') @app.route('/redfish/v1/')
@returns_json @api_utils.returns_json
def root_resource(): def root_resource():
return flask.render_template('root.json') return flask.render_template('root.json')
@app.route('/redfish/v1/Chassis') @app.route('/redfish/v1/Chassis')
@returns_json @api_utils.returns_json
def chassis_collection_resource(): def chassis_collection_resource():
app.logger.debug('Serving chassis list') app.logger.debug('Serving chassis list')
@ -236,7 +193,7 @@ def chassis_collection_resource():
@app.route('/redfish/v1/Chassis/<identity>', methods=['GET', 'PATCH']) @app.route('/redfish/v1/Chassis/<identity>', methods=['GET', 'PATCH'])
@returns_json @api_utils.returns_json
def chassis_resource(identity): def chassis_resource(identity):
chassis = app.chassis chassis = app.chassis
@ -288,7 +245,7 @@ def chassis_resource(identity):
@app.route('/redfish/v1/Chassis/<identity>/Thermal', methods=['GET']) @app.route('/redfish/v1/Chassis/<identity>/Thermal', methods=['GET'])
@returns_json @api_utils.returns_json
def thermal_resource(identity): def thermal_resource(identity):
chassis = app.chassis chassis = app.chassis
@ -312,7 +269,7 @@ def thermal_resource(identity):
@app.route('/redfish/v1/Managers') @app.route('/redfish/v1/Managers')
@returns_json @api_utils.returns_json
def manager_collection_resource(): def manager_collection_resource():
app.logger.debug('Serving managers list') app.logger.debug('Serving managers list')
@ -335,7 +292,7 @@ def jsonify(obj_type, obj_version, obj):
@app.route('/redfish/v1/Managers/<identity>', methods=['GET']) @app.route('/redfish/v1/Managers/<identity>', methods=['GET'])
@returns_json @api_utils.returns_json
def manager_resource(identity): def manager_resource(identity):
app.logger.debug('Serving resources for manager "%s"', identity) app.logger.debug('Serving resources for manager "%s"', identity)
@ -381,179 +338,11 @@ def manager_resource(identity):
}) })
@app.route('/redfish/v1/Managers/<identity>/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/<identity>/VirtualMedia/<device>',
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/<identity>/VirtualMedia/<device>',
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/<identity>/VirtualMedia/<device>/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/<identity>/VirtualMedia/<device>/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/<identity>/VirtualMedia/<device>'
'/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 '<empty>', 'ins': inserted})
return '', 204
@app.route('/redfish/v1/Managers/<identity>/VirtualMedia/<device>'
'/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') @app.route('/redfish/v1/Systems')
@returns_json @api_utils.returns_json
def system_collection_resource(): def system_collection_resource():
systems = [system for system in app.systems.systems 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') app.logger.debug('Serving systems list')
@ -562,8 +351,8 @@ def system_collection_resource():
@app.route('/redfish/v1/Systems/<identity>', methods=['GET', 'PATCH']) @app.route('/redfish/v1/Systems/<identity>', methods=['GET', 'PATCH'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def system_resource(identity): def system_resource(identity):
if flask.request.method == 'GET': if flask.request.method == 'GET':
@ -629,8 +418,8 @@ def system_resource(identity):
@app.route('/redfish/v1/Systems/<identity>/EthernetInterfaces', @app.route('/redfish/v1/Systems/<identity>/EthernetInterfaces',
methods=['GET']) methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def ethernet_interfaces_collection(identity): def ethernet_interfaces_collection(identity):
nics = app.systems.get_nics(identity) nics = app.systems.get_nics(identity)
@ -641,8 +430,8 @@ def ethernet_interfaces_collection(identity):
@app.route('/redfish/v1/Systems/<identity>/EthernetInterfaces/<nic_id>', @app.route('/redfish/v1/Systems/<identity>/EthernetInterfaces/<nic_id>',
methods=['GET']) methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def ethernet_interface(identity, nic_id): def ethernet_interface(identity, nic_id):
nics = app.systems.get_nics(identity) nics = app.systems.get_nics(identity)
@ -656,8 +445,8 @@ def ethernet_interface(identity, nic_id):
@app.route('/redfish/v1/Systems/<identity>/Processors', @app.route('/redfish/v1/Systems/<identity>/Processors',
methods=['GET']) methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def processors_collection(identity): def processors_collection(identity):
processors = app.systems.get_processors(identity) processors = app.systems.get_processors(identity)
@ -668,8 +457,8 @@ def processors_collection(identity):
@app.route('/redfish/v1/Systems/<identity>/Processors/<processor_id>', @app.route('/redfish/v1/Systems/<identity>/Processors/<processor_id>',
methods=['GET']) methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def processor(identity, processor_id): def processor(identity, processor_id):
processors = app.systems.get_processors(identity) processors = app.systems.get_processors(identity)
@ -683,8 +472,8 @@ def processor(identity, processor_id):
@app.route('/redfish/v1/Systems/<identity>/Actions/ComputerSystem.Reset', @app.route('/redfish/v1/Systems/<identity>/Actions/ComputerSystem.Reset',
methods=['POST']) methods=['POST'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def system_reset_action(identity): def system_reset_action(identity):
reset_type = flask.request.json.get('ResetType') reset_type = flask.request.json.get('ResetType')
@ -697,8 +486,8 @@ def system_reset_action(identity):
@app.route('/redfish/v1/Systems/<identity>/BIOS', methods=['GET']) @app.route('/redfish/v1/Systems/<identity>/BIOS', methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def bios(identity): def bios(identity):
bios = app.systems.get_bios(identity) bios = app.systems.get_bios(identity)
@ -712,8 +501,8 @@ def bios(identity):
@app.route('/redfish/v1/Systems/<identity>/BIOS/Settings', @app.route('/redfish/v1/Systems/<identity>/BIOS/Settings',
methods=['GET', 'PATCH']) methods=['GET', 'PATCH'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def bios_settings(identity): def bios_settings(identity):
if flask.request.method == 'GET': if flask.request.method == 'GET':
@ -738,8 +527,8 @@ def bios_settings(identity):
@app.route('/redfish/v1/Systems/<identity>/BIOS/Actions/Bios.ResetBios', @app.route('/redfish/v1/Systems/<identity>/BIOS/Actions/Bios.ResetBios',
methods=['POST']) methods=['POST'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def system_reset_bios(identity): def system_reset_bios(identity):
app.systems.reset_bios(identity) app.systems.reset_bios(identity)
@ -750,8 +539,8 @@ def system_reset_bios(identity):
@app.route('/redfish/v1/Systems/<identity>/SimpleStorage', @app.route('/redfish/v1/Systems/<identity>/SimpleStorage',
methods=['GET']) methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def simple_storage_collection(identity): def simple_storage_collection(identity):
simple_storage_controllers = ( simple_storage_controllers = (
app.systems.get_simple_storage_collection(identity)) app.systems.get_simple_storage_collection(identity))
@ -763,8 +552,8 @@ def simple_storage_collection(identity):
@app.route('/redfish/v1/Systems/<identity>/SimpleStorage/<simple_storage_id>', @app.route('/redfish/v1/Systems/<identity>/SimpleStorage/<simple_storage_id>',
methods=['GET']) methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def simple_storage(identity, simple_storage_id): def simple_storage(identity, simple_storage_id):
simple_storage_controllers = ( simple_storage_controllers = (
app.systems.get_simple_storage_collection(identity)) app.systems.get_simple_storage_collection(identity))
@ -779,8 +568,8 @@ def simple_storage(identity, simple_storage_id):
@app.route('/redfish/v1/Systems/<identity>/Storage', @app.route('/redfish/v1/Systems/<identity>/Storage',
methods=['GET']) methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def storage_collection(identity): def storage_collection(identity):
uuid = app.systems.uuid(identity) uuid = app.systems.uuid(identity)
@ -793,8 +582,8 @@ def storage_collection(identity):
@app.route('/redfish/v1/Systems/<identity>/Storage/<storage_id>', @app.route('/redfish/v1/Systems/<identity>/Storage/<storage_id>',
methods=['GET']) methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def storage(identity, storage_id): def storage(identity, storage_id):
uuid = app.systems.uuid(identity) uuid = app.systems.uuid(identity)
storage_col = app.storage.get_storage_col(uuid) storage_col = app.storage.get_storage_col(uuid)
@ -809,8 +598,8 @@ def storage(identity, storage_id):
@app.route('/redfish/v1/Systems/<identity>/Storage/<stg_id>/Drives/<drv_id>', @app.route('/redfish/v1/Systems/<identity>/Storage/<stg_id>/Drives/<drv_id>',
methods=['GET']) methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def drive_resource(identity, stg_id, drv_id): def drive_resource(identity, stg_id, drv_id):
uuid = app.systems.uuid(identity) uuid = app.systems.uuid(identity)
drives = app.drives.get_drives(uuid, stg_id) 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/<identity>/Storage/<storage_id>/Volumes', @app.route('/redfish/v1/Systems/<identity>/Storage/<storage_id>/Volumes',
methods=['GET', 'POST']) methods=['GET', 'POST'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def volumes_collection(identity, storage_id): def volumes_collection(identity, storage_id):
uuid = app.systems.uuid(identity) uuid = app.systems.uuid(identity)
@ -867,8 +656,8 @@ def volumes_collection(identity, storage_id):
@app.route('/redfish/v1/Systems/<identity>/Storage/<stg_id>/Volumes/<vol_id>', @app.route('/redfish/v1/Systems/<identity>/Storage/<stg_id>/Volumes/<vol_id>',
methods=['GET']) methods=['GET'])
@ensure_instance_access @api_utils.ensure_instance_access
@returns_json @api_utils.returns_json
def volume(identity, stg_id, vol_id): def volume(identity, stg_id, vol_id):
uuid = app.systems.uuid(identity) uuid = app.systems.uuid(identity)
vol_col = app.volumes.get_volumes_col(uuid, stg_id) 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') @app.route('/redfish/v1/Registries')
@returns_json @api_utils.returns_json
def registry_file_collection(): def registry_file_collection():
app.logger.debug('Serving 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') @app.route('/redfish/v1/Registries/BiosAttributeRegistry.v1_0_0')
@returns_json @api_utils.returns_json
def bios_attribute_registry_file(): def bios_attribute_registry_file():
app.logger.debug('Serving 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') @app.route('/redfish/v1/Registries/Messages')
@returns_json @api_utils.returns_json
def message_registry_file(): def message_registry_file():
app.logger.debug('Serving 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') @app.route('/redfish/v1/Systems/Bios/BiosRegistry')
@returns_json @api_utils.returns_json
def bios_registry(): def bios_registry():
app.logger.debug('Serving BIOS registry') app.logger.debug('Serving BIOS registry')
@ -922,7 +711,7 @@ def bios_registry():
@app.route('/redfish/v1/Registries/Messages/Registry') @app.route('/redfish/v1/Registries/Messages/Registry')
@returns_json @api_utils.returns_json
def message_registry(): def message_registry():
app.logger.debug('Serving message registry') app.logger.debug('Serving message registry')

View File

@ -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'])

View File

@ -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'))

View File

@ -16,7 +16,6 @@ from unittest import mock
from oslotest import base from oslotest import base
from sushy_tools.emulator import main from sushy_tools.emulator import main
from sushy_tools.emulator.resources import vmedia
from sushy_tools import error from sushy_tools import error
@ -311,33 +310,6 @@ class SystemsTestCase(EmulatorTestCase):
self.assertEqual(500, response.status_code) 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') @patch_resource('systems')
class BiosTestCase(EmulatorTestCase): class BiosTestCase(EmulatorTestCase):
@ -455,159 +427,6 @@ class EthernetInterfacesTestCase(EmulatorTestCase):
self.assertEqual(404, response.status_code) 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') @patch_resource('systems')
class StorageTestCase(EmulatorTestCase): class StorageTestCase(EmulatorTestCase):