From cdf0cbbc5e08a852482d96fa0defc89704571400 Mon Sep 17 00:00:00 2001 From: "Brian C. Lane" Date: Tue, 16 Apr 2019 16:40:40 -0700 Subject: [PATCH] lorax-composer: Add services support to blueprints Add support for enabling and disabling systemd services in the blueprint. It works like this: [customizations.services] enabled = ["sshd", "cockpit.socket", "httpd"] disabled = ["postfix", "telnetd"] They are *added* to any existing settings in the kickstart templates. (cherry picked from commit 1111aee92dccc6fd619c59c88a3e7779d1a4917a) Related: rhbz#1709595 --- src/pylorax/api/compose.py | 73 ++++++++++++++++++++ tests/pylorax/test_compose.py | 123 ++++++++++++++++++++++++++++++++++ tests/pylorax/test_recipes.py | 13 ++++ 3 files changed, 209 insertions(+) diff --git a/src/pylorax/api/compose.py b/src/pylorax/api/compose.py index ffdaeb42..03e10021 100644 --- a/src/pylorax/api/compose.py +++ b/src/pylorax/api/compose.py @@ -327,6 +327,76 @@ def get_firewall_settings(recipe): return settings +def services_cmd(line, settings): + """ Update the services line with additional services to enable/disable + + :param line: The services ... line + :type line: str + :param settings: A dict with the list of services to enable and disable + :type settings: dict + + Using pykickstart to process the line is the best way to make sure it + is parsed correctly, and re-assembled for inclusion into the final kickstart + """ + # Empty services and no additional settings, return an empty string + if not line and not settings["enabled"] and not settings["disabled"]: + return "" + + ks_version = makeVersion() + ks = KickstartParser(ks_version, errorsAreFatal=False, missingIncludeIsFatal=False) + + # Allow passing in a 'default' so that the enable/disable may be applied to it, without + # parsing it and emitting a kickstart error message + if line != "services": + ks.readKickstartFromString(line) + + # Add to any existing services, removing any duplicates + ks.handler.services.enabled = sorted(set(settings["enabled"] + ks.handler.services.enabled)) + ks.handler.services.disabled = sorted(set(settings["disabled"] + ks.handler.services.disabled)) + + # Converting back to a string includes a comment, return just the keyboard line + return str(ks.handler.services).splitlines()[-1] + + +def get_services(recipe): + """Return the customizations.services settings + + :param recipe: The recipe + :type recipe: Recipe object + :returns: A dict of settings + :rtype: dict + """ + settings = {"enabled": [], "disabled": []} + + if "customizations" not in recipe or \ + "services" not in recipe["customizations"]: + return settings + + settings["enabled"] = sorted(recipe["customizations"]["services"].get("enabled", [])) + settings["disabled"] = sorted(recipe["customizations"]["services"].get("disabled", [])) + return settings + + +def get_default_services(recipe): + """Get the default string for services, based on recipe + :param recipe: The recipe + + :type recipe: Recipe object + :returns: string with "services" or "" + :rtype: str + + When no services have been selected we don't need to add anything to the kickstart + so return an empty string. Otherwise return "services" which will be updated with + the settings. + """ + services = get_services(recipe) + + if services["enabled"] or services["disabled"]: + return "services" + else: + return "" + + def customize_ks_template(ks_template, recipe): """ Customize the kickstart template and return it @@ -363,6 +433,9 @@ def customize_ks_template(ks_template, recipe): "firewall": [firewall_cmd, get_firewall_settings(recipe), 'firewall --enabled', True], + "services": [services_cmd, + get_services(recipe), + get_default_services(recipe), True] } found = {} diff --git a/tests/pylorax/test_compose.py b/tests/pylorax/test_compose.py index 1cbec65c..9d2760e1 100644 --- a/tests/pylorax/test_compose.py +++ b/tests/pylorax/test_compose.py @@ -24,6 +24,7 @@ from pylorax.api.compose import add_customizations, compose_types, get_extra_pkg from pylorax.api.compose import timezone_cmd, get_timezone_settings from pylorax.api.compose import lang_cmd, get_languages, keyboard_cmd, get_keyboard_layout from pylorax.api.compose import firewall_cmd, get_firewall_settings +from pylorax.api.compose import services_cmd, get_services, get_default_services from pylorax.api.compose import get_kernel_append, bootloader_append, customize_ks_template from pylorax.api.config import configure, make_dnf_dirs from pylorax.api.dnfbase import get_base_object @@ -448,6 +449,89 @@ disabled = ["telnet"] "enabled": ["ftp", "ntp", "dhcp"], "disabled": ["telnet"]}), "firewall --disabled") + def test_get_services(self): + """Test get_services function""" + blueprint_data = """name = "test-services" +description = "test recipe" +version = "0.0.1" +[customizations.services] + """ + enable_services = """ +enabled = ["sshd", "cockpit.socket", "httpd"] + """ + disable_services = """ +disabled = ["postfix", "telnetd"] + """ + blueprint2_data = blueprint_data + enable_services + blueprint3_data = blueprint_data + disable_services + blueprint4_data = blueprint_data + enable_services + disable_services + + recipe = recipe_from_toml(blueprint_data) + self.assertEqual(get_services(recipe), {'enabled': [], 'disabled': []}) + + recipe = recipe_from_toml(blueprint2_data) + self.assertEqual(get_services(recipe), + {"enabled": ["cockpit.socket", "httpd", "sshd"], "disabled": []}) + + recipe = recipe_from_toml(blueprint3_data) + self.assertEqual(get_services(recipe), + {"enabled": [], "disabled": ["postfix", "telnetd"]}) + + recipe = recipe_from_toml(blueprint4_data) + self.assertEqual(get_services(recipe), + {"enabled": ["cockpit.socket", "httpd", "sshd"], "disabled": ["postfix", "telnetd"]}) + + def test_services_cmd(self): + """Test services_cmd function""" + + self.assertEqual(services_cmd("", {"enabled": [], "disabled": []}), "") + self.assertEqual(services_cmd("", {"enabled": ["cockpit.socket", "httpd", "sshd"], "disabled": []}), + 'services --enabled="cockpit.socket,httpd,sshd"') + self.assertEqual(services_cmd("", {"enabled": [], "disabled": ["postfix", "telnetd"]}), + 'services --disabled="postfix,telnetd"') + self.assertEqual(services_cmd("", {"enabled": ["cockpit.socket", "httpd", "sshd"], + "disabled": ["postfix", "telnetd"]}), + 'services --disabled="postfix,telnetd" --enabled="cockpit.socket,httpd,sshd"') + self.assertEqual(services_cmd("services --enabled=pop3", {"enabled": ["cockpit.socket", "httpd", "sshd"], + "disabled": ["postfix", "telnetd"]}), + 'services --disabled="postfix,telnetd" --enabled="cockpit.socket,httpd,pop3,sshd"') + self.assertEqual(services_cmd("services --disabled=imapd", {"enabled": ["cockpit.socket", "httpd", "sshd"], + "disabled": ["postfix", "telnetd"]}), + 'services --disabled="imapd,postfix,telnetd" --enabled="cockpit.socket,httpd,sshd"') + self.assertEqual(services_cmd("services --enabled=pop3 --disabled=imapd", {"enabled": ["cockpit.socket", "httpd", "sshd"], + "disabled": ["postfix", "telnetd"]}), + 'services --disabled="imapd,postfix,telnetd" --enabled="cockpit.socket,httpd,pop3,sshd"') + + def test_get_default_services(self): + """Test get_default_services function""" + blueprint_data = """name = "test-services" +description = "test recipe" +version = "0.0.1" + +[customizations.services] + """ + enable_services = """ +enabled = ["sshd", "cockpit.socket", "httpd"] + """ + disable_services = """ +disabled = ["postfix", "telnetd"] + """ + blueprint2_data = blueprint_data + enable_services + blueprint3_data = blueprint_data + disable_services + blueprint4_data = blueprint_data + enable_services + disable_services + + recipe = recipe_from_toml(blueprint_data) + self.assertEqual(get_default_services(recipe), "") + + recipe = recipe_from_toml(blueprint2_data) + self.assertEqual(get_default_services(recipe), "services") + + recipe = recipe_from_toml(blueprint3_data) + self.assertEqual(get_default_services(recipe), "services") + + recipe = recipe_from_toml(blueprint4_data) + self.assertEqual(get_default_services(recipe), "services") + def _checkBootloader(self, result, append_str, line_limit=0): """Find the bootloader line and make sure append_str is in it""" # Optionally check to make sure the change is at the top of the template @@ -535,6 +619,27 @@ disabled = ["telnet"] line_num += 1 return False + def _checkServices(self, result, settings, line_limit=0): + """Find the services line and make sure it is as expected""" + # Optionally check to make sure the change is at the top of the template + line_num = 0 + for line in result.splitlines(): + if line.startswith("services"): + # First layout is used twice, so total count should be n+1 + enabled = all([bool(e in line) for e in settings["enabled"]]) + disabled = all([bool(d in line) for d in settings["disabled"]]) + + if enabled and disabled: + if line_limit == 0 or line_num < line_limit: + return True + else: + print("FAILED: services not in the first %d lines of the output" % line_limit) + return False + else: + print("FAILED: %s not matching %s" % (settings, line)) + line_num += 1 + return False + def test_template_defaults(self): """Test that customize_ks_template includes defaults correctly""" blueprint_data = """name = "test-kernel" @@ -554,6 +659,7 @@ version = "*" self.assertTrue(self._checkBootloader(result, "none", line_limit=2)) self.assertEqual(sum([1 for l in result.splitlines() if l.startswith("timezone")]), 1) self.assertTrue(self._checkTimezone(result, {"timezone": "UTC", "ntpservers": []}, line_limit=2)) + self.assertTrue("services" not in result) # Make sure that a kickstart with a bootloader, and no timezone has timezone added to the top result = customize_ks_template("firewall --enabled\nbootloader --location=mbr\n", recipe) @@ -562,6 +668,7 @@ version = "*" self.assertTrue(self._checkBootloader(result, "mbr")) self.assertEqual(sum([1 for l in result.splitlines() if l.startswith("timezone")]), 1) self.assertTrue(self._checkTimezone(result, {"timezone": "UTC", "ntpservers": []}, line_limit=1)) + self.assertTrue("services" not in result) # Make sure that a kickstart with a bootloader and timezone has neither added result = customize_ks_template("firewall --enabled\nbootloader --location=mbr\ntimezone US/Samoa\n", recipe) @@ -570,6 +677,7 @@ version = "*" self.assertTrue(self._checkBootloader(result, "mbr")) self.assertEqual(sum([1 for l in result.splitlines() if l.startswith("timezone")]), 1) self.assertTrue(self._checkTimezone(result, {"timezone": "US/Samoa", "ntpservers": []})) + self.assertTrue("services" not in result) def test_customize_ks_template(self): """Test that customize_ks_template works correctly""" @@ -594,6 +702,10 @@ ports = ["22:tcp", "80:tcp", "imap:tcp", "53:tcp", "53:udp"] [customizations.firewall.services] enabled = ["ftp", "ntp", "dhcp"] disabled = ["telnet"] + +[customizations.services] +enabled = ["sshd", "cockpit.socket", "httpd"] +disabled = ["postfix", "telnetd"] """ tz_dict = {"timezone": "US/Samoa", "ntpservers": ["0.north-america.pool.ntp.org", "1.north-america.pool.ntp.org"]} recipe = recipe_from_toml(blueprint_data) @@ -612,6 +724,10 @@ disabled = ["telnet"] {"ports": ["22:tcp", "80:tcp", "imap:tcp", "53:tcp", "53:udp"], "enabled": ["ftp", "ntp", "dhcp"], "disabled": ["telnet"]}, line_limit=6)) self.assertEqual(sum([1 for l in result.splitlines() if l.startswith("firewall")]), 1) + self.assertTrue(self._checkServices(result, + {"enabled": ["cockpit.socket", "httpd", "sshd"], "disabled": ["postfix", "telnetd"]}, + line_limit=8)) + self.assertEqual(sum([1 for l in result.splitlines() if l.startswith("services")]), 1) # Test against a kickstart with a bootloader line result = customize_ks_template("firewall --enabled\nbootloader --location=mbr\n", recipe) @@ -664,6 +780,13 @@ disabled = ["telnet"] if sum([1 for l in result.splitlines() if l.startswith("firewall")]) != 1: errors.append(("firewall for compose_type %s failed: More than 1 entry" % compose_type, result)) + if not self._checkServices(result, + {"enabled": ["cockpit.socket", "httpd", "sshd"], + "disabled": ["postfix", "telnetd"]}): + errors.append(("services for compose_type %s failed" % compose_type, result)) + if sum([1 for l in result.splitlines() if l.startswith("services")]) != 1: + errors.append(("services for compose_type %s failed: More than 1 entry" % compose_type, result)) + # Print the bad results for e, r in errors: print("%s:\n%s\n\n" % (e, r)) diff --git a/tests/pylorax/test_recipes.py b/tests/pylorax/test_recipes.py index f4643076..cc9905c3 100644 --- a/tests/pylorax/test_recipes.py +++ b/tests/pylorax/test_recipes.py @@ -544,6 +544,19 @@ disabled = ["telnet"] self.assertEqual(ks.handler.firewall.services, ["ftp", "ntp", "dhcp"]) self.assertEqual(ks.handler.firewall.remove_services, ["telnet"]) + def test_services(self): + blueprint_data = """name = "test-services" +description = "test recipe" +version = "0.0.1" + +[customizations.services] +enabled = ["sshd", "cockpit.socket", "httpd"] +disabled = ["postfix", "telnetd"] +""" + ks = self._blueprint_to_ks(blueprint_data) + self.assertEqual(sorted(ks.handler.services.enabled), ["cockpit.socket", "httpd", "sshd"]) + self.assertEqual(sorted(ks.handler.services.disabled), ["postfix", "telnetd"]) + def test_user(self): blueprint_data = """name = "test-user" description = "test recipe"