Source code for cobbler.cobbler_collections.systems

"""
Copyright 2008-2009, Red Hat, Inc and Others
Michael DeHaan <michael.dehaan AT gmail>

This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 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 General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
02110-1301  USA
"""
from typing import TYPE_CHECKING, Dict, Set

from cobbler.cobbler_collections import collection
from cobbler.items import system as system
from cobbler import utils
from cobbler.cexceptions import CX

if TYPE_CHECKING:
    from cobbler.cobbler_collections.manager import CollectionManager


[docs] class Systems(collection.Collection): """ Systems are hostnames/MACs/IP names and the associated profile they belong to. """
[docs] @staticmethod def collection_type() -> str: return "system"
[docs] @staticmethod def collection_types() -> str: return "systems"
def __init__(self, collection_mgr: "CollectionManager"): """ Constructor. :param collection_mgr: The collection manager to resolve all information with. """ super().__init__(collection_mgr) self.indexes: Dict[str, Dict[str, str]] = { "uid": {}, "mac_address": {}, "ip_address": {}, "ipv6_address": {}, "dns_name": {}, } settings = self.api.settings() self.disabled_indexes: Dict[str, bool] = { "mac_address": settings.allow_duplicate_macs, "ip_address": settings.allow_duplicate_ips, "ipv6_address": settings.allow_duplicate_ips, "dns_name": settings.allow_duplicate_hostnames, }
[docs] def factory_produce(self, api, item_dict): """ Return a System forged from item_dict :param api: TODO :param item_dict: TODO :returns: TODO """ return system.System(api, **item_dict)
[docs] def remove(self, name: str, with_delete: bool = True, with_sync: bool = True, with_triggers: bool = True, recursive: bool = False): """ Remove element named 'name' from the collection :raises CX: In case the name of the object was not given. """ obj = self.find(name=name) if obj is None: raise CX("cannot delete an object that does not exist: %s" % name) if with_delete: if with_triggers: utils.run_triggers(self.api, obj, "/var/lib/cobbler/triggers/delete/system/pre/*", []) if with_sync: lite_sync = self.api.get_sync() lite_sync.remove_single_system(name) self.lock.acquire() try: self.remove_from_indexes(obj) del self.listing[name] finally: self.lock.release() self.collection_mgr.serialize_delete(self, obj) if with_delete: if with_triggers: utils.run_triggers(self.api, obj, "/var/lib/cobbler/triggers/delete/system/post/*", []) utils.run_triggers(self.api, obj, "/var/lib/cobbler/triggers/change/*", [])
[docs] def add_to_indexes(self, ref: system.System) -> None: """ Add indexes for the system. :param ref: The reference to the system whose indexes are updated. """ super().add_to_indexes(ref) if not ref.inmemory: return for indx_key, indx_val in self.indexes.items(): if indx_key == "uid" or self.disabled_indexes[indx_key]: continue for interface in ref.interfaces.values(): if hasattr(interface, indx_key): secondary_key = getattr(interface, indx_key) if secondary_key is not None and secondary_key != "": indx_val[secondary_key] = ref.name
[docs] def update_interface_index_value( self, interface: system.NetworkInterface, attribute_name: str, old_value: str, new_value: str, ) -> None: """ TODO :param interface: TODO :param attribute_name: TODO :param old_value: TODO :param new_value: TODO """ if ( interface.system_name in self.listing and not self.disabled_indexes[attribute_name] and interface in self.listing[interface.system_name].interfaces.values() ): indx_dict = self.indexes[attribute_name] with self.lock: if ( old_value != "" and old_value in indx_dict and indx_dict[old_value] == interface.system_name ): del indx_dict[old_value] if new_value != "": indx_dict[new_value] = interface.system_name
[docs] def update_interfaces_indexes( self, ref: system.System, new_ifaces: Dict[str, system.NetworkInterface] ) -> None: """ Update interfaces indexes for the system. :param ref: The reference to the system whose interfaces indexes are updated. :param new_ifaces: The new interfaces. """ if ref.name not in self.listing: return for indx_key, indx_val in self.indexes.items(): if indx_key == "uid" or self.disabled_indexes[indx_key]: continue old_ifaces = ref.interfaces old_values: Set[str] = { getattr(x, indx_key) for x in old_ifaces.values() if hasattr(x, indx_key) and getattr(x, indx_key) != "" } new_values: Set[str] = { getattr(x, indx_key) for x in new_ifaces.values() if hasattr(x, indx_key) and getattr(x, indx_key) != "" } with self.lock: for value in old_values - new_values: del indx_val[value] for value in new_values - old_values: indx_val[value] = ref.name
[docs] def update_interface_indexes( self, ref: system.System, iface_name: str, new_iface: system.NetworkInterface ) -> None: """ Update interface indexes for the system. :param ref: The reference to the system whose interfaces indexes are updated. :param iface_name: The new interface name. :param new_iface: The new interface. """ self.update_interfaces_indexes( ref, {**ref.interfaces, **{iface_name: new_iface}} )
[docs] def remove_from_indexes(self, ref: system.System) -> None: """ Remove index keys for the system. :param ref: The reference to the system whose index keys are removed. """ if not ref.inmemory: return super().remove_from_indexes(ref) for indx_key, indx_val in self.indexes.items(): if indx_key == "uid" or self.disabled_indexes[indx_key]: continue for interface in ref.interfaces.values(): if hasattr(interface, indx_key): indx_val.pop(getattr(interface, indx_key), None)
[docs] def remove_interface_from_indexes(self, ref: system.System, name: str) -> None: """ Remove index keys for the system interface. :param ref: The reference to the system whose index keys are removed. :param name: The reference to the system whose index keys are removed. """ if not ref.inmemory or name not in ref.interfaces: return interface = ref.interfaces[name] with self.lock: for indx_key, indx_val in self.indexes.items(): if indx_key == "uid" or self.disabled_indexes[indx_key]: continue if hasattr(interface, indx_key): indx_val.pop(getattr(interface, indx_key), None)