#!/usr/bin/env python3 # -*- coding: utf-8 -*- # # === This file is part of Calamares - === # # Copyright 2016, Artoo # Copyright 2017, Philip Müller # Copyright 2018, Artoo # Copyright 2018-2019, Adriaan de Groot # # Calamares 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 3 of the License, or # (at your option) any later version. # # Calamares 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 Calamares. If not, see . import libcalamares from libcalamares.utils import target_env_call, warning from os.path import exists, join import gettext _ = gettext.translation("calamares-python", localedir=libcalamares.utils.gettext_path(), languages=libcalamares.utils.gettext_languages(), fallback=True).gettext def pretty_name(): return _("Configure OpenRC services") class OpenrcController: """ This is the openrc service controller. All of its state comes from global storage and the job configuration at initialization time. """ def __init__(self): self.root = libcalamares.globalstorage.value('rootMountPoint') # Translate the entries in the config to the actions passed to rc-config self.services = dict() self.services["add"] = libcalamares.job.configuration.get('services', []) self.services["del"] = libcalamares.job.configuration.get('disable', []) self.initdDir = libcalamares.job.configuration['initdDir'] self.runlevelsDir = libcalamares.job.configuration['runlevelsDir'] def make_failure_description(self, state, name, runlevel): """ Returns a generic "could not " failure message, specialized for the action @p state and the specific service @p name in @p runlevel. """ if state == "add": description = _("Cannot add service {name!s} to run-level {level!s}.") elif state == "del": description = _("Cannot remove service {name!s} from run-level {level!s}.") else: description = _("Unknown service-action {arg!s} for service {name!s} in run-level {level!s}.") return description.format(arg=state, name=name, level=runlevel) def update(self, state): """ Call rc-update for each service listed in services for the given @p state. rc-update is called with @p state as the command as well. """ for svc in self.services.get(state, []): if isinstance(svc, str): name = svc runlevel = "default" mandatory = False else: name = svc["name"] runlevel = svc.get("runlevel", "default") mandatory = svc.get("mandatory", False) service_path = self.root + self.initdDir + "/" + name runlevel_path = self.root + self.runlevelsDir + "/" + runlevel if exists(service_path): if exists(runlevel_path): ec = target_env_call(["rc-update", state, name, runlevel]) if ec != 0: warning("Cannot {} service {} to {}".format(state, name, runlevel)) warning("rc-update returned error code {!s}".format(ec)) if mandatory: title = _("Cannot modify service") diagnostic = _("rc-update {arg!s} call in chroot returned error code {num!s}.").format(arg=state, num=ec) return (title, self.make_failure_description(state, name, runlevel) + " " + diagnostic ) else: warning("Target runlevel {} does not exist for {}.".format(runlevel, name)) if mandatory: title = _("Target runlevel does not exist") diagnostic = _("The path for runlevel {level!s} is {path!s}, which does not exist.").format(level=runlevel, path=runlevel_path) return (title, self.make_failure_description(state, name, runlevel) + " " + diagnostic ) else: warning("Target service {} does not exist in {}.".format(name, self.initdDir)) if mandatory: title = _("Target service does not exist") diagnostic = _("The path for service {name!s} is {path!s}, which does not exist.").format(name=name, path=service_path) return (title, self.make_failure_description(state, name, runlevel) + " " + diagnostic ) def run(self): """Run the controller """ for state in ("add", "del"): r = self.update(state) if r is not None: return r def run(): """ Setup services """ return OpenrcController().run()