Source code for ironic.objects.firmware

# coding=utf-8
#
#
#    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 oslo_versionedobjects import base as object_base

from ironic.db import api as dbapi
from ironic.objects import base
from ironic.objects import fields as object_fields


[docs] @base.IronicObjectRegistry.register class FirmwareComponent(base.IronicObject): # Version 1.0: Initial version VERSION = '1.0' dbapi = dbapi.get_instance() fields = { 'id': object_fields.IntegerField(), 'node_id': object_fields.IntegerField(nullable=False), 'component': object_fields.StringField(nullable=False), 'initial_version': object_fields.StringField(nullable=False), 'current_version': object_fields.StringField(nullable=True), 'last_version_flashed': object_fields.StringField(nullable=True), }
[docs] def create(self, context=None): """Create a Firmware record in the DB. :param context: Security context. :raises: NodeNotFound if the node is not found. :raises: FirmwareComponentAlreadyExists if the record already exists. """ values = self.do_version_changes_for_db() # Note(iurygregory): We ensure that when creating we will be setting # initial_version to the current_version we got from the BMC. values['initial_version'] = values['current_version'] db_fwcmp = self.dbapi.create_firmware_component(values) self._from_db_object(self._context, self, db_fwcmp)
[docs] def save(self, context=None): """Save updates to this Firmware Component. Updates will be made column by column based on the result of self.what_changed() :param context: Security context. NOTE: This should only be used internally by the indirection_api. Unfortunately, RPC requires context as the first argument, even though we don't use it. A context should be set when instantiating the object, e.g.: FirmwareComponent(context) :raises: NodeNotFound if the node id is not found. :raises: FirmwareComponentNotFound if the component is not found. """ # NOTE(iurygregory): some fields shouldn't be updated, like # 'initial_version', 'id', 'node_id', 'component' # filter them out or raise an Error? updates = self.do_version_changes_for_db() up_fwcmp = self.dbapi.update_firmware_component( self.node_id, self.component, updates) self._from_db_object(self._context, self, up_fwcmp)
[docs] @classmethod def get(cls, context, node_id, name): """Get a FirmwareComponent based on its node_id and name. :param context: Security context. :param node_id: The node id. :param name: The Firmware Component name. :raises: NodeNotFound if the node id is not found. :raises: FirmwareComponentNotFound if the Firmware Component name is not found. :returns: A :class:'FirmwareComponent' object. """ db_fw_cmp = cls.dbapi.get_firmware_component(node_id, name) fw_cmp = cls._from_db_object(context, cls(), db_fw_cmp) return fw_cmp
[docs] @base.IronicObjectRegistry.register class FirmwareComponentList(base.IronicObjectListBase, base.IronicObject): # Version 1.0: Initial version VERSION = '1.0' dbapi = dbapi.get_instance() fields = { 'objects': object_fields.ListOfObjectsField('FirmwareComponent'), }
[docs] @classmethod def get_by_node_id(cls, context, node_id): """Get FirmwareComponent based on node_id. :param context: Security context. :param node_id: The node id. :raises: NodeNotFound if the node is not found. :return: A list of FirmwareComponent objects. """ node_fw_components = cls.dbapi.get_firmware_component_list(node_id) return object_base.obj_make_list( context, cls(), FirmwareComponent, node_fw_components)
[docs] @classmethod def sync_firmware_components(cls, context, node_id, components): """Returns a list of create/update components. This method sync with the 'firmware_information' database table and sorts three lists - create / update / unchanged components. :param context: Security context. :param node_id: The node id. :param components: List of FirmwareComponents. :returns: A 3-tuple of lists of Firmware Components to be created, updated and unchanged. """ create_list = [] update_list = [] unchanged_list = [] current_components_dict = {} current_components = cls.get_by_node_id(context, node_id) for cmp in current_components: current_components_dict[cmp.component] = { 'initial_version': cmp.initial_version, 'current_version': cmp.current_version, 'last_version_flashed': cmp.last_version_flashed, } for cmp in components: if cmp['component'] in current_components_dict: values = current_components_dict[cmp['component']] if values.get('last_version_flashed') is None: lvf_changed = False cv_changed = cmp['current_version'] \ != values.get('current_version') else: lvf_changed = cmp['current_version'] \ != values.get('last_version_flashed') cv_changed = cmp['current_version'] \ != values.get('current_version') if cv_changed or lvf_changed: update_list.append(cmp) else: unchanged_list.append(cmp) else: create_list.append(cmp) return (create_list, update_list, unchanged_list)