# # Copyright (C) 2017 Red Hat, Inc. # # 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, see . # import configparser import grp import os from pylorax.sysutils import joinpaths class ComposerConfig(configparser.SafeConfigParser): def get_default(self, section, option, default): try: return self.get(section, option) except configparser.Error: return default def configure(conf_file="/etc/lorax/composer.conf", root_dir="/", test_config=False): """lorax-composer configuration :param conf_file: Path to the config file overriding the default settings :type conf_file: str :param root_dir: Directory to prepend to paths, defaults to / :type root_dir: str :param test_config: Set to True to skip reading conf_file :type test_config: bool """ conf = ComposerConfig() # set defaults conf.add_section("composer") conf.set("composer", "share_dir", os.path.realpath(joinpaths(root_dir, "/usr/share/lorax/"))) conf.set("composer", "lib_dir", os.path.realpath(joinpaths(root_dir, "/var/lib/lorax/composer/"))) conf.set("composer", "dnf_conf", os.path.realpath(joinpaths(root_dir, "/var/tmp/composer/dnf.conf"))) conf.set("composer", "dnf_root", os.path.realpath(joinpaths(root_dir, "/var/tmp/composer/dnf/root/"))) conf.set("composer", "repo_dir", os.path.realpath(joinpaths(root_dir, "/var/tmp/composer/repos.d/"))) conf.set("composer", "cache_dir", os.path.realpath(joinpaths(root_dir, "/var/tmp/composer/cache/"))) conf.set("composer", "tmp", os.path.realpath(joinpaths(root_dir, "/var/tmp/"))) conf.add_section("users") conf.set("users", "root", "1") # Enable all available repo files by default conf.add_section("repos") conf.set("repos", "use_system_repos", "1") conf.set("repos", "enabled", "*") if not test_config: # read the config file if os.path.isfile(conf_file): conf.read(conf_file) return conf def make_dnf_dirs(conf): """Make any missing dnf directories :param conf: The configuration to use :type conf: ComposerConfig :returns: None """ for p in ["dnf_conf", "repo_dir", "cache_dir", "dnf_root"]: p_dir = os.path.dirname(conf.get("composer", p)) if not os.path.exists(p_dir): os.makedirs(p_dir) def make_queue_dirs(conf, gid): """Make any missing queue directories :param conf: The configuration to use :type conf: ComposerConfig :param gid: Group ID that has access to the queue directories :type gid: int :returns: list of errors :rtype: list of str """ errors = [] lib_dir = conf.get("composer", "lib_dir") for p in ["queue/run", "queue/new", "results"]: p_dir = joinpaths(lib_dir, p) if not os.path.exists(p_dir): orig_umask = os.umask(0) os.makedirs(p_dir, 0o771) os.chown(p_dir, 0, gid) os.umask(orig_umask) else: p_stat = os.stat(p_dir) if p_stat.st_mode & 0o006 != 0: errors.append("Incorrect permissions on %s, no o+rw permissions are allowed." % p_dir) if p_stat.st_gid != gid or p_stat.st_uid != 0: gr_name = grp.getgrgid(gid).gr_name errors.append("%s should be owned by root:%s" % (p_dir, gr_name)) return errors