Source code for linux.base_objects

import json
import shlex

from base_objects import VMEndpoint, AbstractUnixEndpoint

from firewheel.control.experiment_graph import (
    IncorrectConflictHandlerError,
    require_class,
)


[docs] @require_class(VMEndpoint) @require_class(AbstractUnixEndpoint) class LinuxHost: """ A class with functionality common to all Linux Hosts. """ def __init__(self, name=None): """ Sets a few of the basic options for new Linux-based VMs. Arguments: name (str): The name of the VM. Raises: RuntimeError: If no name is given to the VM (or provided earlier). """ self.type = "host" self.name = getattr(self, "name", name) if not self.name: raise RuntimeError("LinuxHost needs a name!") self.set_hostname() self.add_root_profiles()
[docs] def set_hostname(self, start_time=-250): """ Wrapper to run the vm_resource that sets the hostname of the VM Args: start_time (int, optional): The start time to configure the VM's hostname (default=-250) """ self.run_executable(start_time, "set_hostname.sh", self.name, vm_resource=True)
[docs] def change_password(self, start_time, username, password): """ Wrapper to change passwords for Linux VMs. Arguments: start_time (int): The schedule time to configure the VM's password. username (str): The username whose password should change. password (str): The new password. """ self.run_executable( start_time, "chpasswd.sh", "{} {}".format(username, password), vm_resource=True, )
[docs] def cleanup(self, start_time=1): """ Clean up the FIREWHEEL artifacts from the VM (e.g. the ``/var/launch`` folder). Arguments: start_time (int): The start time to remove the artifacts. Default is 1. """ self.run_executable(start_time, "/bin/rm", "-rf /var/launch")
[docs] def increase_ulimit(self, fd_limit=102400): """ This helps users adjust common `ulimit <https://ss64.com/bash/ulimit.html>`_ parameters that typically impact experiments. Depending on how a process was created, the limit which impacts that process differs therefore, it is sometimes difficult to discern how to properly increase limits. Users whom are logged in via the PAM module might see a limit set by ``limits.conf`` whereas system services have a different limit altogether. This method is a best effort to increase the limits in various locations. Currently, we only increase the number of file descriptors open, but this method can be extended in the future for other ``ulimit`` parameters. .. seealso:: - https://wiki.archlinux.org/title/Limits.conf - https://www.cyberciti.biz/faq/linux-increase-the-maximum-number-of-open-files/ - https://pubs.opengroup.org/onlinepubs/009695399/functions/getrlimit.html - https://www.scivision.dev/platform-independent-ulimit-number-of-open-files-increase/ - https://man.archlinux.org/man/systemd-system.conf.5 Arguments: fd_limit (int): The maximum number of open file descriptors. Defaults to 102400. """ start_time = -900 # Set the default nofile ulimit self.run_executable( start_time, "set_ulimit.sh", arguments=f"{fd_limit}", vm_resource=True, )
[docs] def add_root_profiles(self): """ Adds default ssh keys, .bashrc, .vimrc, etc. for the ``root`` user. """ # root self.drop_file(-249, "/root/combined_profiles.tgz", "combined_profiles.tgz") self.run_executable( -248, "chown", "-R root:root /root/combined_profiles.tgz", vm_resource=False ) self.run_executable( -247, "tar", "--no-same-owner -C /root/ -xf /root/combined_profiles.tgz" ) self.run_executable(-246, "rm", "-f /root/combined_profiles.tgz")
[docs] def configure_ips(self, start_time=-200): """ Configure the IP addresses of the VM Args: start_time (int): The start time to configure the VM's hostname (default=-200) Returns: bool: True if successful, False otherwise. """ self.interfaces = getattr(self, "interfaces", None) if not self.interfaces: return False try: nameservers = self.dns_nameservers if isinstance(nameservers, list): nameservers = " ".join(nameservers) except AttributeError: nameservers = "" config = "" for iface in self.interfaces.interfaces: if ( "mac" in iface and "address" in iface and iface["address"] and "netmask" in iface and iface["netmask"] ): config += "%s %s %s %s %s" % ( iface["switch"].name, iface["mac"], str(iface["address"]), str(iface["netmask"]), str(iface["network"].prefixlen), ) # Add default gateway if there is one gateway = getattr(self, "default_gateway", None) if gateway and not iface["control_network"]: config += f" {gateway}" config += "\n" if not config: return config = f"{nameservers}\n{config}" self.add_vm_resource(start_time, "configure_ips.sh", config) return True
[docs] def unpack_tar( self, time, archive, options="-xzf", directory=None, vm_resource=False ): """ Unpack the tar archive. This unpacks the tar archive, optionally into a specified directory. By default, the archive will be unpacked using the ``'-xzf'`` set of options, reading from the file given as the ``archive`` argument. Other option combinations can be passed directly to the tar executable via the ``options`` parameter, or indirectly via the other method parameters. Args: time (int): The schedule time (positive or negative) at which the tarball will be unpacked. archive (str, pathlib.Path): The location of the archive file to be unpacked on the VM (or, if ``vm_resource`` is :py:data:`True`, the name of the VM resource). Unless ``vm_resource`` is :py:data:`True`, it is safest to specify the absolute path of the archive on the VM. options (str): The set of options to be passed to the tar executable. This string must begin with ``'-x'`` and end with ``'f'`` (or ``'-f'``) since this method only performs extractions of named archives. directory (str or pathlib.Path, optional): A directory where the archiving utility will move before unpacking the archive. Specifying a directory is recommended when running with ``vm_resource`` set to :py:data:`True`. vm_resource (bool, optional): A flag indicating whether the archive is a VM resource and needs to be loaded onto the VM before it is unpacked. Defaults to :py:data:`False`. Raises: ValueError: If the provided options are unsupported. """ if not options.startswith("-x"): raise ValueError( "The `options` parameter must begin with '-x' since this method " "only supports archive extraction." ) if not options.endswith("f"): raise ValueError( "The `options` parameter must end with 'f' (or '-f') since this " "method requires that an archive file be specified for extraction." ) tar_options = shlex.split(options) if directory: # Prevent duplicate `directory` options lest the kwarg be silently ignored if any(option in options for option in ["-C", "--directory"]): raise ValueError( "The directory option was provided via both the `options` " "parameter and the `directory` parameter; use only one." ) tar_options = ["-C", str(directory), *tar_options] tar_arguments = [*tar_options, str(archive)] exec_vm_resource = self.run_executable(time, "tar", arguments=tar_arguments) # If the archive is a known VMR, load it onto the VM if vm_resource: exec_vm_resource.add_file(archive, archive)
[docs] def configure_ip_conflict_handler(entry_name, _decorator_value, _current_instance_value): """ The conflict handler for functions overwritten in LinuxNetplanHost that are also implemented in LinuxHost, i.e. the ``configure_ips`` function. Args: entry_name (str): A string describing the attribute that has a conflict _decorator_value (any): The value of the attribute from the class that it is trying to be decorated by _current_instance_value (any): The current value of the conflicting attribute Returns: function: A function to be used as the ``configure_ips`` function for Netplan-enabled hosts Raises: IncorrectConflictHandlerError: If the conflicting function is not ``"configure_ips"``. """ if entry_name == "configure_ips": return LinuxNetplanHost.configure_ips raise IncorrectConflictHandlerError
[docs] @require_class(LinuxHost, conflict_handler=configure_ip_conflict_handler) class LinuxNetplanHost: """ A class that implements functionality for Linux machines that use `Netplan <https://netplan.io/>`__ """ def __init__(self): """ Nothing to do here """
[docs] def configure_ips(self, start_time=-200): """ Configure the IP addresses of the VM using netplan Args: start_time (int): The start time to configure the VM's hostname (default=-200) Returns: bool: True if successful, False otherwise. """ self.interfaces = getattr(self, "interfaces", None) if not self.interfaces: return False try: nameservers = self.dns_nameservers if isinstance(nameservers, str): nameservers = nameservers.split(" ") except AttributeError: nameservers = [] ethernets = {} macs = [] for iface in self.interfaces.interfaces: if "mac" in iface and "address" in iface and iface["address"]: ip_addr = str(iface["address"]) prefixlen = str(iface["network"].prefixlen) address = f"{ip_addr}/{prefixlen}" mac = iface["mac"] macs.append(mac) ethernets[mac] = { "addresses": [address], "nameservers": {"addresses": nameservers}, } if not iface["control_network"] and hasattr(self, "default_gateway"): ethernets[mac]["gateway4"] = str(self.default_gateway) config = {"network": {"ethernets": ethernets, "version": 2}} if len(ethernets) == 0: return # Even though it uses YAML, we use JSON (since all JSON is valid YAML) # for ease of editing in other scripts if other settings need to be # applied self.drop_content( start_time - 1, "/etc/netplan/firewheel.yaml", json.dumps(config) ) macs_str = " ".join(macs) self.run_executable( start_time, "set_netplan_interfaces.sh", arguments=f'"{macs_str}"', vm_resource=True, ) return True