Source code for pywbem_mock._namespaceprovider

#
# (C) Copyright 2020 InovaDevelopment.com
#
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2.1 of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful, but
# WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
# License along with this program; if not, write to the Free Software
# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
#
# Author: Karl  Schopmeyer <inovadevelopment.com>
#

"""
This module implements a user-defined provider for the class CIM_Namespace

This  provider manages creation and deletion of the instances of the
CIM_Namespace class and limits that activity to the server environment interop
classname.  The CIM_Namespace class is a DMTF defined class that
defines an instance for each namespace in the server environment and allows
the creation and deletion of server CIM namespaces to be controlled through
creation and deletion of instances of  the CIM_Namespace class.
"""

from nocaselist import NocaseList

from pywbem import CIMError, CIMInstance, CIM_ERR_NOT_SUPPORTED, \
    CIM_ERR_INVALID_PARAMETER
from pywbem._nocasedict import NocaseDict
from pywbem._utils import _format

from ._instancewriteprovider import InstanceWriteProvider
from .config import OBJECTMANAGERNAME, SYSTEMNAME, SYSTEMCREATIONCLASSNAME, \
    OBJECTMANAGERCREATIONCLASSNAME

__all__ = ['CIMNamespaceProvider']

# CIM class name of the namespace class implemented in this provider
NAMESPACE_CLASSNAME = 'CIM_Namespace'


[docs]class CIMNamespaceProvider(InstanceWriteProvider): # pylint: disable=line-too-long """ Implements the user defined provider for the class CIM_Namespace. This provider provides the create, modify, and delete methods for adding an instance of the class CIM_Namspace when a namespace is created or deleted in a pywbem mock environment. This class and the instances of this class only exist in the WBEM server Interop namespace per DMTF definitions. This class __init__ saves the cimrepository variable used by methods in this class and by methods of its superclasses (i.e. InstanceWriteProvider). The provider defines the class level attribute `provider_classnames` (CIM_Namespace) This provider presumes that an Interop namespace has been created before the provider object is constructed and fails the constructor if there is not interop_namespace """ # noqa: E501 # pylint: enable=line-too-long # This class level attribute must exist to define the CIM classname(es) . # for which this provider is responsible. #: provider_classnames (:term:`string`): #: The classname for this provider provider_classnames = NAMESPACE_CLASSNAME def __init__(self, cimrepository): """ Parameters: cimrepository (:class:`~pywbem_mock.BaseRepository` or subclass): Defines the CIM repository to be used by the provider. """ super(CIMNamespaceProvider, self).__init__(cimrepository) if not self.find_interop_namespace(): raise CIMError( CIM_ERR_INVALID_PARAMETER, _format("Cannot create namespace provider (for class {0}): " "No Interop namespace exists in the CIM repository. " "Valid Interop namespaces are: {1}", NAMESPACE_CLASSNAME, ", ".join(self.interop_namespace_names))) self.installed = False # test if provider previously installed.
[docs] def __repr__(self): return _format( "CIMNamespaceProvider(" "provider_type={s.provider_type}, " "provider_classnames={s.provider_classnames})", s=self)
[docs] def CreateInstance(self, namespace, new_instance): """ Create an instance of the CIM_Namespace class in an Interop namespace of the CIM repository, and if not yet existing create the new namespace in the CIM repository. See `~pywbem_mock.InstanceWriteProvider.CreateInstance` for documentation of validation and description of input parameters, noting extra conditions for this provider as described below: Parameters: namespace (:term:`string`): Must be a valid Interop namespace. new_instance (:class:`~pywbem.CIMInstance`): The following applies regarding its properties: * 'Name' property: This property is required since it defines the name of the new namespace to be created. * 'CreationClassName' property: This property is required and its value must match the class name of the new instance. Raises: :exc:`~pywbem.CIMError`: (CIM_ERR_INVALID_PARAMETER) if namespace is not the Interop namespace in the CIM repository or the Name property does not exist or the other properties cannot be added to the instance. """ if not self.is_interop_namespace(namespace): raise CIMError( CIM_ERR_INVALID_PARAMETER, _format( "Cannot create instance of class {0!A} in namespace {1!A}: " "The namespace is not an Interop namespace. " "Valid Interop namespaces are: {2!A}", new_instance.classname, namespace, ", ".join(self.interop_namespace_names))) ccn_pname = 'CreationClassName' name_pname = 'Name' # Validate that required properties are specified in the new instance for pn in [name_pname, ccn_pname]: if pn not in new_instance: raise CIMError( CIM_ERR_INVALID_PARAMETER, _format( "Cannot create instance of class {0!A} in " "namespace {1!A}: " "Missing required property {2!A} in new_instance", new_instance.classname, namespace, pn)) # Get and normalize the new namespace name new_namespace = new_instance[name_pname] # Property value new_namespace = new_namespace.strip('/') # Write it back to the instance in case it was changed new_instance[name_pname] = new_namespace # Validate other property values in the new instance if new_instance[ccn_pname].lower() != new_instance.classname.lower(): raise CIMError( CIM_ERR_INVALID_PARAMETER, _format( "Cannot create instance of class {0!A} in namespace " "{1!A}: Value of property {2|A} in new_instance does not " "match class name but is {3!A}", new_instance.classname, namespace, ccn_pname, new_instance[ccn_pname])) # Create the new namespace in the CIM repository, if needed. # The add_namespace() method will prevent the creation of a second # Interop namespace, raising CIMError(CIM_ERR_ALREADY_EXISTS). if new_namespace not in self.cimrepository.namespaces: self.add_namespace(new_namespace) # Create the CIM instance for the new namespace in the CIM repository, # by delegating to the default provider method. return super(CIMNamespaceProvider, self).CreateInstance( namespace, new_instance)
[docs] def ModifyInstance(self, modified_instance, IncludeQualifiers=None): """ Modification of CIM_Namespace instance not allowed """ raise CIMError( CIM_ERR_NOT_SUPPORTED, _format("Modification of {0} instances is not allowed: " "{1!A}", NAMESPACE_CLASSNAME, modified_instance.path))
[docs] def DeleteInstance(self, InstanceName): """ Delete an instance of the CIM_Namespace class in an Interop namespace of the CIM repository, and in addition delete the namespace represented by it in the CIM repository. See `~pywbem_mock.InstanceWriteProvider.CreateInstance` for documentation of validation and description of input parameters, noting extra conditions for this provider as described below: The namespace to be deleted must be empty and must not be the Interop namespace. Parameters: InstanceName: (:class:`~pywbem.CIMInstanceName`): The keybinding `Name` must exist; it defines the namespace to be deleted. Raises: :exc:`~pywbem.CIMError`: (CIM_ERR_INVALID_PARAMETER) :exc:`~pywbem.CIMError`: (CIM_ERR_NAMESPACE_NOT_EMPTY) """ # The provider dispatcher ensures that provider methods are only called # for the registered classes. # And this provider sets only a single class, not a list. assert InstanceName.classname.lower() == \ CIMNamespaceProvider.provider_classnames.lower() remove_namespace = InstanceName.keybindings['Name'] if self.is_interop_namespace(remove_namespace): raise CIMError( CIM_ERR_INVALID_PARAMETER, _format("Cannot delete instance {0!A} from the CIM repository: " "This instance represents the Interop namespace {1!A} " "which must not be deleted.", InstanceName, remove_namespace)) # Delete the namespace from the CIM repository. # This call verifies that the namespace is empty and raises # CIMError(CIM_ERR_NAMESPACE_NOT_EMPTY) if not empty. self.remove_namespace(remove_namespace) # Delete the instance from the CIM repository instance_store = self.cimrepository.get_instance_store( InstanceName.namespace) instance_store.delete(InstanceName)
[docs] def post_register_setup(self, conn): """ Method called by FakedWBEMConnection.register_provider to complete initialization of this provider. This method is called after the required classes are installed in the cim_repository This is necessary because pywbem_mock does not allow user-defined providers for the instance read operations such as EnumerateInstances so the instance for each namespace must be exist in the repository. This method: 1. Inserts instances of CIM_Namespace for every namespace in the CIM repository. """ assert self.installed is False self.installed = True # Temp code to validate class installed. Will generate exception # if class does not exist. interop_namespace = conn.find_interop_namespace() provider_classname = self.provider_classnames conn.GetClass(provider_classname, interop_namespace) # Create instance CIM_Namespace for each existing namespace insts = conn.EnumerateInstances('CIM_Namespace', namespace=conn.find_interop_namespace(), LocalOnly=False, DeepInheritance=False) # List of namespaces for which there are CIM instances inst_ns_list = NocaseList( [inst['Name'] for inst in insts if 'Name' in inst]) klass = conn.GetClass(provider_classname, interop_namespace, LocalOnly=False, IncludeQualifiers=True, IncludeClassOrigin=True) # Set up any CIM_Namespace instances that are still missing given the # namespaces that exist in the CIM repository. for ns in conn.cimrepository.namespaces: if ns not in inst_ns_list: self.create_cimnamespace_instance(conn, ns, interop_namespace, klass)
[docs] @staticmethod def create_cimnamespace_instance(conn, namespace, interop_namespace, klass): """ Build and execute CreateInstance for an instance of CIM_Namespace using the `namespace` parameter as the value of the name property in the instance. The instance is created in the `interop_namespace` Parameters: namespace (:term:`string`): Namespace that this instance defines. interop_namespace (:term:`string`): Interop namespace for this environment. klass (:class:`~pywbem.CIMClass`): The CIM class CIM_Namespace which is used to create the instance. Raises: :exc:`~pywbem.CIMError`: For errors encountered with CreateInstance """ properties = NocaseDict( [('Name', namespace), ('CreationClassName', klass.classname), ('ObjectManagerName', OBJECTMANAGERNAME), ('ObjectManagerCreationClassName', OBJECTMANAGERCREATIONCLASSNAME), ('SystemName', SYSTEMNAME), ('SystemCreationClassName', SYSTEMCREATIONCLASSNAME)]) new_instance = CIMInstance.from_class(klass, property_values=properties) conn.CreateInstance(new_instance, interop_namespace)