From a9c5581aa94595665b001d8d827726d4aaf60462 Mon Sep 17 00:00:00 2001 From: "Brian C. Lane" Date: Fri, 12 Apr 2019 12:23:20 -0700 Subject: [PATCH] lorax-composer: Add locale support to blueprints You can now set the keyboard layout and language. Eg. [customizations.locale] languages = ["en_CA.utf8", "en_HK.utf8"] keyboard = "de (dvorak)" Existing entries in the kickstart templates are replaced with the new ones. If there are no entries then it will default to 'keyboard us' and 'lang en_US.UTF-8' Includes tests, and leaves the existing keyboard and lang entries in the templates with a note that they can be replaced by the blueprint. (cherry picked from commit e5a8700bdfe980a22c3cdbf44a1e6dc264c7c2ef) Related: rhbz#1718473 --- docs/lorax-composer.rst | 15 ++-- share/composer/ami.ks | 1 + share/composer/ext4-filesystem.ks | 1 + share/composer/live-iso.ks | 1 + share/composer/openstack.ks | 1 + share/composer/partitioned-disk.ks | 1 + share/composer/qcow2.ks | 1 + share/composer/tar.ks | 1 + share/composer/vhd.ks | 1 + share/composer/vmdk.ks | 1 + src/pylorax/api/compose.py | 85 ++++++++++++++++++++ tests/pylorax/test_compose.py | 125 +++++++++++++++++++++++++++++ tests/pylorax/test_recipes.py | 54 +++++++++++++ 13 files changed, 281 insertions(+), 7 deletions(-) diff --git a/docs/lorax-composer.rst b/docs/lorax-composer.rst index 9553d761..38047f38 100644 --- a/docs/lorax-composer.rst +++ b/docs/lorax-composer.rst @@ -254,23 +254,24 @@ cannot be overridden because they are required to boot in the selected environme timezone will be updated to the one selected in the blueprint. -[[customizations.locale]] +[customizations.locale] +*********************** Customize the locale settings for the system:: - [[customizations.locale]] - language = "en_US.UTF-8" + [customizations.locale] + languages = ["en_US.UTF-8"] keyboard = "us" -The values supported by ``language`` can be listed by running ``localectl list-locales`` from +The values supported by ``languages`` can be listed by running ``localectl list-locales`` from the command line. The values supported by ``keyboard`` can be listed by running ``localectl list-keymaps`` from the command line. -Multiple locale and keyboard sections can be used. The first one becomes the -primary, and the others are added as secondary. One or the other of ``language`` -or ``keyboard`` must be included (or both). +Multiple languages can be added. The first one becomes the +primary, and the others are added as secondary. One or the other of ``languages`` +or ``keyboard`` must be included (or both) in the section. [customizations.firewall] diff --git a/share/composer/ami.ks b/share/composer/ami.ks index c1d2f4b1..fa8b2d83 100644 --- a/share/composer/ami.ks +++ b/share/composer/ami.ks @@ -11,6 +11,7 @@ firewall --enabled network --bootproto=dhcp --onboot=on --activate # System authorization information auth --useshadow --enablemd5 +# NOTE: keyboard and lang can be replaced by blueprint customizations.locale settings # System keyboard keyboard --xlayouts=us --vckeymap=us # System language diff --git a/share/composer/ext4-filesystem.ks b/share/composer/ext4-filesystem.ks index 8b74bc84..2d6868cb 100644 --- a/share/composer/ext4-filesystem.ks +++ b/share/composer/ext4-filesystem.ks @@ -6,6 +6,7 @@ firewall --enabled # NOTE: The root account is locked by default # Network information network --bootproto=dhcp --onboot=on --activate +# NOTE: keyboard and lang can be replaced by blueprint customizations.locale settings # System keyboard keyboard --xlayouts=us --vckeymap=us # System language diff --git a/share/composer/live-iso.ks b/share/composer/live-iso.ks index e8a3db04..b6c352db 100644 --- a/share/composer/live-iso.ks +++ b/share/composer/live-iso.ks @@ -9,6 +9,7 @@ xconfig --startxonboot rootpw --plaintext removethispw # Network information network --bootproto=dhcp --onboot=on --activate +# NOTE: keyboard and lang can be replaced by blueprint customizations.locale settings # System keyboard keyboard --xlayouts=us --vckeymap=us # System language diff --git a/share/composer/openstack.ks b/share/composer/openstack.ks index 55ae137a..d71e0910 100644 --- a/share/composer/openstack.ks +++ b/share/composer/openstack.ks @@ -6,6 +6,7 @@ firewall --disabled # NOTE: The root account is locked by default # Network information network --bootproto=dhcp --onboot=on --activate +# NOTE: keyboard and lang can be replaced by blueprint customizations.locale settings # System keyboard keyboard --xlayouts=us --vckeymap=us # System language diff --git a/share/composer/partitioned-disk.ks b/share/composer/partitioned-disk.ks index 49b4635a..6e2ffa5c 100644 --- a/share/composer/partitioned-disk.ks +++ b/share/composer/partitioned-disk.ks @@ -6,6 +6,7 @@ firewall --enabled # NOTE: The root account is locked by default # Network information network --bootproto=dhcp --onboot=on --activate +# NOTE: keyboard and lang can be replaced by blueprint customizations.locale settings # System keyboard keyboard --xlayouts=us --vckeymap=us # System language diff --git a/share/composer/qcow2.ks b/share/composer/qcow2.ks index e9639e95..488a2fe6 100644 --- a/share/composer/qcow2.ks +++ b/share/composer/qcow2.ks @@ -6,6 +6,7 @@ firewall --enabled # NOTE: The root account is locked by default # Network information network --bootproto=dhcp --onboot=on --activate +# NOTE: keyboard and lang can be replaced by blueprint customizations.locale settings # System keyboard keyboard --xlayouts=us --vckeymap=us # System language diff --git a/share/composer/tar.ks b/share/composer/tar.ks index 69473686..7c3dd2dc 100644 --- a/share/composer/tar.ks +++ b/share/composer/tar.ks @@ -6,6 +6,7 @@ firewall --enabled # NOTE: The root account is locked by default # Network information network --bootproto=dhcp --onboot=on --activate +# NOTE: keyboard and lang can be replaced by blueprint customizations.locale settings # System keyboard keyboard --xlayouts=us --vckeymap=us # System language diff --git a/share/composer/vhd.ks b/share/composer/vhd.ks index cdef8fca..80ba0eba 100644 --- a/share/composer/vhd.ks +++ b/share/composer/vhd.ks @@ -9,6 +9,7 @@ firewall --enabled # NOTE: The root account is locked by default # Network information network --bootproto=dhcp --onboot=on --activate +# NOTE: keyboard and lang can be replaced by blueprint customizations.locale settings # System keyboard keyboard --xlayouts=us --vckeymap=us # System language diff --git a/share/composer/vmdk.ks b/share/composer/vmdk.ks index 4dc410b6..f9db6b51 100644 --- a/share/composer/vmdk.ks +++ b/share/composer/vmdk.ks @@ -6,6 +6,7 @@ firewall --enabled # NOTE: The root account is locked by default # Network information network --bootproto=dhcp --onboot=on --activate +# NOTE: keyboard and lang can be replaced by blueprint customizations.locale settings # System keyboard keyboard --xlayouts=us --vckeymap=us # System language diff --git a/src/pylorax/api/compose.py b/src/pylorax/api/compose.py index f6b35b8d..bb48cd65 100644 --- a/src/pylorax/api/compose.py +++ b/src/pylorax/api/compose.py @@ -208,6 +208,85 @@ def get_timezone_settings(recipe): return recipe["customizations"]["timezone"] +def lang_cmd(line, languages): + """ Update the lang line with the languages + + :param line: The lang ... line + :type line: str + :param settings: The list of languages + :type settings: list + + 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 + """ + ks_version = makeVersion() + ks = KickstartParser(ks_version, errorsAreFatal=False, missingIncludeIsFatal=False) + ks.readKickstartFromString(line) + + if languages: + ks.handler.lang.lang = languages[0] + + if len(languages) > 1: + ks.handler.lang.addsupport = languages[1:] + + # Converting back to a string includes a comment, return just the lang line + return str(ks.handler.lang).splitlines()[-1] + + +def get_languages(recipe): + """Return the customizations.locale.languages list + + :param recipe: The recipe + :type recipe: Recipe object + :returns: list of language strings + :rtype: list + """ + if "customizations" not in recipe or \ + "locale" not in recipe["customizations"] or \ + "languages" not in recipe["customizations"]["locale"]: + return [] + return recipe["customizations"]["locale"]["languages"] + + +def keyboard_cmd(line, layout): + """ Update the keyboard line with the layout + + :param line: The keyboard ... line + :type line: str + :param settings: The keyboard layout + :type settings: str + + 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 + """ + ks_version = makeVersion() + ks = KickstartParser(ks_version, errorsAreFatal=False, missingIncludeIsFatal=False) + ks.readKickstartFromString(line) + + if layout: + ks.handler.keyboard.keyboard = layout + ks.handler.keyboard.vc_keymap = "" + ks.handler.keyboard.x_layouts = [] + + # Converting back to a string includes a comment, return just the keyboard line + return str(ks.handler.keyboard).splitlines()[-1] + + +def get_keyboard_layout(recipe): + """Return the customizations.locale.keyboard list + + :param recipe: The recipe + :type recipe: Recipe object + :returns: The keyboard layout string + :rtype: str + """ + if "customizations" not in recipe or \ + "locale" not in recipe["customizations"] or \ + "keyboard" not in recipe["customizations"]["locale"]: + return [] + return recipe["customizations"]["locale"]["keyboard"] + + def customize_ks_template(ks_template, recipe): """ Customize the kickstart template and return it @@ -235,6 +314,12 @@ def customize_ks_template(ks_template, recipe): "timezone": [timezone_cmd, get_timezone_settings(recipe), 'timezone UTC', False], + "lang": [lang_cmd, + get_languages(recipe), + 'lang en_US.UTF-8', True], + "keyboard": [keyboard_cmd, + get_keyboard_layout(recipe), + 'keyboard --xlayouts us --vckeymap us', True], } found = {} diff --git a/tests/pylorax/test_compose.py b/tests/pylorax/test_compose.py index 7ba7eec6..5f4e05b0 100644 --- a/tests/pylorax/test_compose.py +++ b/tests/pylorax/test_compose.py @@ -19,6 +19,7 @@ import unittest from pylorax.api.compose import add_customizations, compose_types 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 get_kernel_append, bootloader_append, customize_ks_template from pylorax.api.recipes import recipe_from_toml from pylorax.sysutils import joinpaths @@ -288,6 +289,78 @@ ntpservers = ["0.north-america.pool.ntp.org", "1.north-america.pool.ntp.org"] {"timezone": "US/Samoa", "ntpservers": ["0.pool.ntp.org", "1.pool.ntp.org"]}), 'timezone US/Samoa --ntpservers=0.pool.ntp.org,1.pool.ntp.org') + def test_get_languages(self): + """Test get_languages function""" + blueprint_data = """name = "test-locale" +description = "test recipe" +version = "0.0.1" + """ + blueprint2_data = blueprint_data + """ +[customizations.locale] +languages = ["en_CA.utf8", "en_HK.utf8"] +""" + blueprint3_data = blueprint_data + """ +[customizations.locale] +keyboard = "de (dvorak)" +languages = ["en_CA.utf8", "en_HK.utf8"] +""" + recipe = recipe_from_toml(blueprint_data) + self.assertEqual(get_languages(recipe), []) + + recipe = recipe_from_toml(blueprint2_data) + self.assertEqual(get_languages(recipe), ["en_CA.utf8", "en_HK.utf8"]) + + recipe = recipe_from_toml(blueprint3_data) + self.assertEqual(get_languages(recipe), ["en_CA.utf8", "en_HK.utf8"]) + + def test_lang_cmd(self): + """Test lang_cmd function""" + + self.assertEqual(lang_cmd("lang en_CA.utf8", {}), 'lang en_CA.utf8') + self.assertEqual(lang_cmd("lang en_US.utf8", ["en_HK.utf8"]), + 'lang en_HK.utf8') + self.assertEqual(lang_cmd("lang en_US.utf8", ["en_CA.utf8", "en_HK.utf8"]), + 'lang en_CA.utf8 --addsupport=en_HK.utf8') + + self.assertEqual(lang_cmd("lang --addsupport en_US.utf8 en_CA.utf8", + ["en_CA.utf8", "en_HK.utf8", "en_GB.utf8"]), + 'lang en_CA.utf8 --addsupport=en_HK.utf8,en_GB.utf8') + + def test_get_keyboard_layout(self): + """Test get_keyboard_layout function""" + blueprint_data = """name = "test-locale" +description = "test recipe" +version = "0.0.1" + """ + blueprint2_data = blueprint_data + """ +[customizations.locale] +keyboard = "de (dvorak)" +""" + blueprint3_data = blueprint_data + """ +[customizations.locale] +keyboard = "de (dvorak)" +languages = ["en_CA.utf8", "en_HK.utf8"] +""" + recipe = recipe_from_toml(blueprint_data) + self.assertEqual(get_keyboard_layout(recipe), []) + + recipe = recipe_from_toml(blueprint2_data) + self.assertEqual(get_keyboard_layout(recipe), "de (dvorak)") + + recipe = recipe_from_toml(blueprint3_data) + self.assertEqual(get_keyboard_layout(recipe), "de (dvorak)") + + def test_keyboard_cmd(self): + """Test lang_cmd function""" + + self.assertEqual(keyboard_cmd("keyboard us", {}), "keyboard 'us'") + self.assertEqual(keyboard_cmd("keyboard us", "de (dvorak)"), + "keyboard 'de (dvorak)'") + + self.assertEqual(keyboard_cmd("keyboard --vckeymap=us --xlayouts=us,gb", + "de (dvorak)"), + "keyboard 'de (dvorak)'") + 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 @@ -319,6 +392,40 @@ ntpservers = ["0.north-america.pool.ntp.org", "1.north-america.pool.ntp.org"] line_num += 1 return False + def _checkLang(self, result, locales, line_limit=0): + """Find the lang 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("lang"): + if all([True for n in locales if n in line]): + if line_limit == 0 or line_num < line_limit: + return True + else: + print("FAILED: lang not in the first %d lines of the output" % line_limit) + return False + else: + print("FAILED: %s not matching %s" % (locales, line)) + line_num += 1 + return False + + def _checkKeyboard(self, result, layout, line_limit=0): + """Find the keyboard 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("keyboard"): + if layout in line: + if line_limit == 0 or line_num < line_limit: + return True + else: + print("FAILED: keyboard not in the first %d lines of the output" % line_limit) + return False + else: + print("FAILED: %s not matching %s" % (layout, line)) + line_num += 1 + return False + def test_template_defaults(self): """Test that customize_ks_template includes defaults correctly""" blueprint_data = """name = "test-kernel" @@ -367,6 +474,10 @@ append="nosmt=force" [customizations.timezone] timezone = "US/Samoa" ntpservers = ["0.north-america.pool.ntp.org", "1.north-america.pool.ntp.org"] + +[customizations.locale] +keyboard = "de (dvorak)" +languages = ["en_CA.utf8", "en_HK.utf8"] """ tz_dict = {"timezone": "US/Samoa", "ntpservers": ["0.north-america.pool.ntp.org", "1.north-america.pool.ntp.org"]} recipe = recipe_from_toml(blueprint_data) @@ -377,6 +488,10 @@ ntpservers = ["0.north-america.pool.ntp.org", "1.north-america.pool.ntp.org"] self.assertEqual(sum([1 for l in result.splitlines() if l.startswith("bootloader")]), 1) self.assertTrue(self._checkTimezone(result, tz_dict, line_limit=2)) self.assertEqual(sum([1 for l in result.splitlines() if l.startswith("timezone")]), 1) + self.assertTrue(self._checkLang(result, ["en_CA.utf8", "en_HK.utf8"], line_limit=4)) + self.assertEqual(sum([1 for l in result.splitlines() if l.startswith("lang")]), 1) + self.assertTrue(self._checkKeyboard(result, "de (dvorak)", line_limit=4)) + self.assertEqual(sum([1 for l in result.splitlines() if l.startswith("keyboard")]), 1) # Test against a kickstart with a bootloader line result = customize_ks_template("firewall --enabled\nbootloader --location=mbr\n", recipe) @@ -407,6 +522,16 @@ ntpservers = ["0.north-america.pool.ntp.org", "1.north-america.pool.ntp.org"] if sum([1 for l in result.splitlines() if l.startswith("timezone")]) != 1: errors.append(("timezone for compose_type %s failed: More than 1 entry" % compose_type, result)) + if not self._checkLang(result, ["en_CA.utf8", "en_HK.utf8"]): + errors.append(("lang for compose_type %s failed" % compose_type, result)) + if sum([1 for l in result.splitlines() if l.startswith("lang")]) != 1: + errors.append(("lang for compose_type %s failed: More than 1 entry" % compose_type, result)) + + if not self._checkKeyboard(result, "de (dvorak)"): + errors.append(("keyboard for compose_type %s failed" % compose_type, result)) + if sum([1 for l in result.splitlines() if l.startswith("keyboard")]) != 1: + errors.append(("keyboard 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 a04187c1..b55652dc 100644 --- a/tests/pylorax/test_recipes.py +++ b/tests/pylorax/test_recipes.py @@ -440,6 +440,60 @@ ntpservers = ["1.north-america.pool.ntp.org"] self.assertEqual(ks.handler.timezone.timezone, "US/Samoa") self.assertEqual(ks.handler.timezone.ntpservers, ["1.north-america.pool.ntp.org"]) + def test_locale_languages(self): + blueprint_data = """name = "test-locale" +description = "test recipe" +version = "0.0.1" +""" + blueprint2_data = blueprint_data + """ +[customizations.locale] +languages = ["en_CA.utf8"] +""" + blueprint3_data = blueprint_data + """ +[customizations.locale] +languages = ["en_CA.utf8", "en_HK.utf8"] +""" + ks = self._blueprint_to_ks(blueprint2_data) + self.assertEqual(ks.handler.lang.lang, "en_CA.utf8") + self.assertEqual(ks.handler.lang.addsupport, []) + + ks = self._blueprint_to_ks(blueprint3_data) + self.assertEqual(ks.handler.lang.lang, "en_CA.utf8") + self.assertEqual(ks.handler.lang.addsupport, ["en_HK.utf8"]) + + def test_locale_keyboard(self): + blueprint_data = """name = "test-locale" +description = "test recipe" +version = "0.0.1" +""" + blueprint2_data = blueprint_data + """ +[customizations.locale] +keyboard = "us" +""" + blueprint3_data = blueprint_data + """ +[customizations.locale] +keyboard = "de (dvorak)" +""" + ks = self._blueprint_to_ks(blueprint2_data) + self.assertEqual(ks.handler.keyboard.keyboard, "us") + + ks = self._blueprint_to_ks(blueprint3_data) + self.assertEqual(ks.handler.keyboard.keyboard, "de (dvorak)") + + def test_locale(self): + blueprint_data = """name = "test-locale" +description = "test recipe" +version = "0.0.1" + +[customizations.locale] +keyboard = "de (dvorak)" +languages = ["en_CA.utf8", "en_HK.utf8"] +""" + ks = self._blueprint_to_ks(blueprint_data) + self.assertEqual(ks.handler.keyboard.keyboard, "de (dvorak)") + self.assertEqual(ks.handler.lang.lang, "en_CA.utf8") + self.assertEqual(ks.handler.lang.addsupport, ["en_HK.utf8"]) + def test_user(self): blueprint_data = """name = "test-user" description = "test recipe"